| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  | from __future__ import annotations | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | import copy | 
					
						
							| 
									
										
										
										
											2021-02-20 02:30:55 +01:00
										 |  |  | import functools | 
					
						
							| 
									
										
										
										
											2023-01-15 11:10:26 -06:00
										 |  |  | import json | 
					
						
							|  |  |  | import logging | 
					
						
							|  |  |  | import random | 
					
						
							|  |  |  | import secrets | 
					
						
							|  |  |  | import typing  # this can go away when Python 3.8 support is dropped | 
					
						
							|  |  |  | from argparse import Namespace | 
					
						
							| 
									
										
										
										
											2020-05-10 19:27:13 +10:00
										 |  |  | from collections import OrderedDict, Counter, deque | 
					
						
							| 
									
										
										
										
											2023-01-15 11:10:26 -06:00
										 |  |  | from enum import unique, IntEnum, IntFlag | 
					
						
							| 
									
										
										
										
											2022-05-11 13:05:53 -05:00
										 |  |  | from typing import List, Dict, Optional, Set, Iterable, Union, Any, Tuple, TypedDict, Callable, NamedTuple | 
					
						
							| 
									
										
										
										
											2020-07-14 07:01:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-15 11:10:26 -06:00
										 |  |  | import NetUtils | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  | import Options | 
					
						
							| 
									
										
										
										
											2021-09-13 02:01:15 +02:00
										 |  |  | import Utils | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  | class Group(TypedDict, total=False): | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  |     name: str | 
					
						
							|  |  |  |     game: str | 
					
						
							|  |  |  |     world: auto_world | 
					
						
							|  |  |  |     players: Set[int] | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |     item_pool: Set[str] | 
					
						
							|  |  |  |     replacement_items: Dict[int, Optional[str]] | 
					
						
							| 
									
										
										
										
											2022-05-15 07:41:11 -07:00
										 |  |  |     local_items: Set[str] | 
					
						
							|  |  |  |     non_local_items: Set[str] | 
					
						
							| 
									
										
										
										
											2022-12-07 06:37:47 +01:00
										 |  |  |     link_replacement: bool | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-14 07:01:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-24 05:38:56 +02:00
										 |  |  | class MultiWorld(): | 
					
						
							| 
									
										
										
										
											2020-04-10 20:54:18 +02:00
										 |  |  |     debug_types = False | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  |     player_name: Dict[int, str] | 
					
						
							| 
									
										
										
										
											2021-07-21 18:08:15 +02:00
										 |  |  |     _region_cache: Dict[int, Dict[str, Region]] | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     difficulty_requirements: dict | 
					
						
							|  |  |  |     required_medallions: dict | 
					
						
							| 
									
										
										
										
											2020-10-07 19:51:46 +02:00
										 |  |  |     dark_room_logic: Dict[int, str] | 
					
						
							|  |  |  |     restrict_dungeon_item_on_boss: Dict[int, bool] | 
					
						
							| 
									
										
										
										
											2021-01-02 22:41:03 +01:00
										 |  |  |     plando_texts: List[Dict[str, str]] | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  |     plando_items: List[List[Dict[str, Any]]] | 
					
						
							| 
									
										
										
										
											2021-07-21 18:08:15 +02:00
										 |  |  |     plando_connections: List | 
					
						
							| 
									
										
										
										
											2022-04-28 09:03:44 -07:00
										 |  |  |     worlds: Dict[int, auto_world] | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  |     groups: Dict[int, Group] | 
					
						
							| 
									
										
										
										
											2022-09-28 14:54:10 -07:00
										 |  |  |     regions: List[Region] | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  |     itempool: List[Item] | 
					
						
							| 
									
										
										
										
											2021-07-20 21:19:53 +02:00
										 |  |  |     is_race: bool = False | 
					
						
							| 
									
										
										
										
											2021-10-10 16:50:01 +02:00
										 |  |  |     precollected_items: Dict[int, List[Item]] | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |     state: CollectionState | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-17 17:25:59 +01:00
										 |  |  |     plando_options: PlandoOptions | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  |     accessibility: Dict[int, Options.Accessibility] | 
					
						
							| 
									
										
										
										
											2022-11-16 10:32:33 -06:00
										 |  |  |     early_items: Dict[int, Dict[str, int]] | 
					
						
							|  |  |  |     local_early_items: Dict[int, Dict[str, int]] | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  |     local_items: Dict[int, Options.LocalItems] | 
					
						
							|  |  |  |     non_local_items: Dict[int, Options.NonLocalItems] | 
					
						
							|  |  |  |     progression_balancing: Dict[int, Options.ProgressionBalancing] | 
					
						
							| 
									
										
										
										
											2022-04-07 10:42:30 -07:00
										 |  |  |     completion_condition: Dict[int, Callable[[CollectionState], bool]] | 
					
						
							| 
									
										
										
										
											2022-09-30 04:58:19 +02:00
										 |  |  |     indirect_connections: Dict[Region, Set[Entrance]] | 
					
						
							| 
									
										
										
										
											2022-09-28 14:54:10 -07:00
										 |  |  |     exclude_locations: Dict[int, Options.ExcludeLocations] | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-18 09:54:41 -07:00
										 |  |  |     game: Dict[int, str] | 
					
						
							| 
									
										
										
										
											2022-09-30 04:58:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-10 06:36:06 +02:00
										 |  |  |     class AttributeProxy(): | 
					
						
							|  |  |  |         def __init__(self, rule): | 
					
						
							|  |  |  |             self.rule = rule | 
					
						
							| 
									
										
										
										
											2021-04-10 18:45:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-10 06:36:06 +02:00
										 |  |  |         def __getitem__(self, player) -> bool: | 
					
						
							|  |  |  |             return self.rule(player) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-14 08:38:02 +01:00
										 |  |  |     def __init__(self, players: int): | 
					
						
							| 
									
										
										
										
											2021-03-14 22:59:41 +01:00
										 |  |  |         self.random = random.Random()  # world-local random state is saved for multiple generations running concurrently | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         self.players = players | 
					
						
							| 
									
										
										
										
											2022-01-30 13:57:12 +01:00
										 |  |  |         self.player_types = {player: NetUtils.SlotType.player for player in self.player_ids} | 
					
						
							| 
									
										
										
										
											2021-03-22 13:14:19 -07:00
										 |  |  |         self.glitch_triforce = False | 
					
						
							| 
									
										
										
										
											2021-03-14 08:38:02 +01:00
										 |  |  |         self.algorithm = 'balanced' | 
					
						
							| 
									
										
										
										
											2021-08-29 16:02:28 +02:00
										 |  |  |         self.dungeons: Dict[Tuple[str, int], Dungeon] = {} | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  |         self.groups = {} | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         self.regions = [] | 
					
						
							| 
									
										
										
										
											2018-02-17 18:38:54 -05:00
										 |  |  |         self.shops = [] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         self.itempool = [] | 
					
						
							| 
									
										
										
										
											2017-05-20 14:03:15 +02:00
										 |  |  |         self.seed = None | 
					
						
							| 
									
										
										
										
											2021-05-16 00:21:00 +02:00
										 |  |  |         self.seed_name: str = "Unavailable" | 
					
						
							| 
									
										
										
										
											2021-10-10 16:50:01 +02:00
										 |  |  |         self.precollected_items = {player: [] for player in self.player_ids} | 
					
						
							| 
									
										
										
										
											2019-01-20 01:01:02 -06:00
										 |  |  |         self._cached_entrances = None | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         self._cached_locations = None | 
					
						
							|  |  |  |         self._entrance_cache = {} | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  |         self._location_cache: Dict[Tuple[str, int], Location] = {} | 
					
						
							| 
									
										
										
										
											2017-06-04 16:15:59 +02:00
										 |  |  |         self.required_locations = [] | 
					
						
							| 
									
										
										
										
											2017-06-03 21:27:34 +02:00
										 |  |  |         self.light_world_light_cone = False | 
					
						
							| 
									
										
										
										
											2017-06-03 15:46:05 +02:00
										 |  |  |         self.dark_world_light_cone = False | 
					
						
							| 
									
										
										
										
											2018-01-21 20:43:44 -06:00
										 |  |  |         self.rupoor_cost = 10 | 
					
						
							| 
									
										
										
										
											2017-08-01 18:58:42 +02:00
										 |  |  |         self.aga_randomness = True | 
					
						
							| 
									
										
										
										
											2017-06-04 13:10:22 +02:00
										 |  |  |         self.lock_aga_door_in_escape = False | 
					
						
							| 
									
										
										
										
											2018-09-22 22:51:54 -04:00
										 |  |  |         self.save_and_quit_from_boss = True | 
					
						
							| 
									
										
										
										
											2021-03-14 08:38:02 +01:00
										 |  |  |         self.custom = False | 
					
						
							|  |  |  |         self.customitemarray = [] | 
					
						
							|  |  |  |         self.shuffle_ganon = True | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  |         self.spoiler = Spoiler(self) | 
					
						
							| 
									
										
										
										
											2022-11-16 10:32:33 -06:00
										 |  |  |         self.early_items = {player: {} for player in self.player_ids} | 
					
						
							|  |  |  |         self.local_early_items = {player: {} for player in self.player_ids} | 
					
						
							| 
									
										
										
										
											2022-09-30 04:58:19 +02:00
										 |  |  |         self.indirect_connections = {} | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |         self.fix_trock_doors = self.AttributeProxy( | 
					
						
							|  |  |  |             lambda player: self.shuffle[player] != 'vanilla' or self.mode[player] == 'inverted') | 
					
						
							|  |  |  |         self.fix_skullwoods_exit = self.AttributeProxy( | 
					
						
							|  |  |  |             lambda player: self.shuffle[player] not in ['vanilla', 'simple', 'restricted', 'dungeonssimple']) | 
					
						
							|  |  |  |         self.fix_palaceofdarkness_exit = self.AttributeProxy( | 
					
						
							|  |  |  |             lambda player: self.shuffle[player] not in ['vanilla', 'simple', 'restricted', 'dungeonssimple']) | 
					
						
							|  |  |  |         self.fix_trock_exit = self.AttributeProxy( | 
					
						
							|  |  |  |             lambda player: self.shuffle[player] not in ['vanilla', 'simple', 'restricted', 'dungeonssimple']) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-17 21:09:33 +01:00
										 |  |  |         for player in range(1, players + 1): | 
					
						
							|  |  |  |             def set_player_attr(attr, val): | 
					
						
							|  |  |  |                 self.__dict__.setdefault(attr, {})[player] = val | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-10 03:03:46 +02:00
										 |  |  |             set_player_attr('tech_tree_layout_prerequisites', {}) | 
					
						
							| 
									
										
										
										
											2019-12-17 21:09:33 +01:00
										 |  |  |             set_player_attr('_region_cache', {}) | 
					
						
							| 
									
										
										
										
											2021-03-14 08:38:02 +01:00
										 |  |  |             set_player_attr('shuffle', "vanilla") | 
					
						
							|  |  |  |             set_player_attr('logic', "noglitches") | 
					
						
							|  |  |  |             set_player_attr('mode', 'open') | 
					
						
							|  |  |  |             set_player_attr('difficulty', 'normal') | 
					
						
							|  |  |  |             set_player_attr('item_functionality', 'normal') | 
					
						
							|  |  |  |             set_player_attr('timer', False) | 
					
						
							|  |  |  |             set_player_attr('goal', 'ganon') | 
					
						
							| 
									
										
										
										
											2019-12-17 21:09:33 +01:00
										 |  |  |             set_player_attr('required_medallions', ['Ether', 'Quake']) | 
					
						
							|  |  |  |             set_player_attr('swamp_patch_required', False) | 
					
						
							|  |  |  |             set_player_attr('powder_patch_required', False) | 
					
						
							|  |  |  |             set_player_attr('ganon_at_pyramid', True) | 
					
						
							|  |  |  |             set_player_attr('ganonstower_vanilla', True) | 
					
						
							| 
									
										
										
										
											2019-12-18 20:45:51 +01:00
										 |  |  |             set_player_attr('can_access_trock_eyebridge', None) | 
					
						
							|  |  |  |             set_player_attr('can_access_trock_front', None) | 
					
						
							|  |  |  |             set_player_attr('can_access_trock_big_chest', None) | 
					
						
							|  |  |  |             set_player_attr('can_access_trock_middle', None) | 
					
						
							|  |  |  |             set_player_attr('fix_fake_world', True) | 
					
						
							| 
									
										
										
										
											2019-12-17 21:09:33 +01:00
										 |  |  |             set_player_attr('difficulty_requirements', None) | 
					
						
							|  |  |  |             set_player_attr('boss_shuffle', 'none') | 
					
						
							|  |  |  |             set_player_attr('enemy_health', 'default') | 
					
						
							|  |  |  |             set_player_attr('enemy_damage', 'default') | 
					
						
							| 
									
										
										
										
											2021-11-03 05:34:11 +00:00
										 |  |  |             set_player_attr('beemizer_total_chance', 0) | 
					
						
							|  |  |  |             set_player_attr('beemizer_trap_chance', 0) | 
					
						
							| 
									
										
										
										
											2019-12-17 21:09:33 +01:00
										 |  |  |             set_player_attr('escape_assist', []) | 
					
						
							| 
									
										
										
										
											2019-12-21 10:42:59 +01:00
										 |  |  |             set_player_attr('treasure_hunt_icon', 'Triforce Piece') | 
					
						
							|  |  |  |             set_player_attr('treasure_hunt_count', 0) | 
					
						
							| 
									
										
										
										
											2020-03-04 13:55:03 +01:00
										 |  |  |             set_player_attr('clock_mode', False) | 
					
						
							| 
									
										
										
										
											2020-10-28 16:20:59 -07:00
										 |  |  |             set_player_attr('countdown_start_time', 10) | 
					
						
							|  |  |  |             set_player_attr('red_clock_time', -2) | 
					
						
							|  |  |  |             set_player_attr('blue_clock_time', 2) | 
					
						
							|  |  |  |             set_player_attr('green_clock_time', 4) | 
					
						
							| 
									
										
										
										
											2020-02-02 21:52:57 -05:00
										 |  |  |             set_player_attr('can_take_damage', True) | 
					
						
							| 
									
										
										
										
											2020-06-17 01:02:54 -07:00
										 |  |  |             set_player_attr('triforce_pieces_available', 30) | 
					
						
							| 
									
										
										
										
											2020-06-07 15:22:24 +02:00
										 |  |  |             set_player_attr('triforce_pieces_required', 20) | 
					
						
							| 
									
										
										
										
											2020-08-23 15:03:06 +02:00
										 |  |  |             set_player_attr('shop_shuffle', 'off') | 
					
						
							| 
									
										
										
										
											2020-09-20 04:35:45 +02:00
										 |  |  |             set_player_attr('shuffle_prizes', "g") | 
					
						
							| 
									
										
										
										
											2020-10-06 13:22:03 -07:00
										 |  |  |             set_player_attr('sprite_pool', []) | 
					
						
							| 
									
										
										
										
											2020-10-07 19:51:46 +02:00
										 |  |  |             set_player_attr('dark_room_logic', "lamp") | 
					
						
							| 
									
										
										
										
											2021-01-02 12:49:43 +01:00
										 |  |  |             set_player_attr('plando_items', []) | 
					
						
							| 
									
										
										
										
											2021-01-02 16:44:58 +01:00
										 |  |  |             set_player_attr('plando_texts', {}) | 
					
						
							| 
									
										
										
										
											2021-01-02 22:41:03 +01:00
										 |  |  |             set_player_attr('plando_connections', []) | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |             set_player_attr('game', "A Link to the Past") | 
					
						
							| 
									
										
										
										
											2021-02-22 11:18:53 +01:00
										 |  |  |             set_player_attr('completion_condition', lambda state: True) | 
					
						
							| 
									
										
										
										
											2021-04-24 01:16:49 +02:00
										 |  |  |         self.custom_data = {} | 
					
						
							| 
									
										
										
										
											2021-06-11 14:22:44 +02:00
										 |  |  |         self.worlds = {} | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |         self.slot_seeds = {} | 
					
						
							| 
									
										
										
										
											2023-01-17 17:25:59 +01:00
										 |  |  |         self.plando_options = PlandoOptions.none | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-20 20:50:32 +01:00
										 |  |  |     def get_all_ids(self) -> Tuple[int, ...]: | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |         return self.player_ids + tuple(self.groups) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  |     def add_group(self, name: str, game: str, players: Set[int] = frozenset()) -> Tuple[int, Group]: | 
					
						
							|  |  |  |         """Create a group with name and return the assigned player ID and group.
 | 
					
						
							|  |  |  |         If a group of this name already exists, the set of players is extended instead of creating a new one."""
 | 
					
						
							|  |  |  |         for group_id, group in self.groups.items(): | 
					
						
							|  |  |  |             if group["name"] == name: | 
					
						
							|  |  |  |                 group["players"] |= players | 
					
						
							|  |  |  |                 return group_id, group | 
					
						
							|  |  |  |         new_id: int = self.players + len(self.groups) + 1 | 
					
						
							| 
									
										
										
										
											2022-02-18 20:29:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  |         self.game[new_id] = game | 
					
						
							|  |  |  |         self.custom_data[new_id] = {} | 
					
						
							|  |  |  |         self.player_types[new_id] = NetUtils.SlotType.group | 
					
						
							| 
									
										
										
										
											2022-02-21 14:59:01 -08:00
										 |  |  |         self._region_cache[new_id] = {} | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  |         world_type = AutoWorld.AutoWorldRegister.world_types[game] | 
					
						
							| 
									
										
										
										
											2022-08-15 16:46:59 -05:00
										 |  |  |         for option_key, option in world_type.option_definitions.items(): | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  |             getattr(self, option_key)[new_id] = option(option.default) | 
					
						
							|  |  |  |         for option_key, option in Options.common_options.items(): | 
					
						
							|  |  |  |             getattr(self, option_key)[new_id] = option(option.default) | 
					
						
							|  |  |  |         for option_key, option in Options.per_game_common_options.items(): | 
					
						
							|  |  |  |             getattr(self, option_key)[new_id] = option(option.default) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.worlds[new_id] = world_type(self, new_id) | 
					
						
							| 
									
										
										
										
											2022-02-22 10:14:26 +01:00
										 |  |  |         self.worlds[new_id].collect_item = classmethod(AutoWorld.World.collect_item).__get__(self.worlds[new_id]) | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  |         self.player_name[new_id] = name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         new_group = self.groups[new_id] = Group(name=name, game=game, players=players, | 
					
						
							|  |  |  |                                                 world=self.worlds[new_id]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return new_id, new_group | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-05 16:55:11 +01:00
										 |  |  |     def get_player_groups(self, player) -> Set[int]: | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  |         return {group_id for group_id, group in self.groups.items() if player in group["players"]} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |     def set_seed(self, seed: Optional[int] = None, secure: bool = False, name: Optional[str] = None): | 
					
						
							|  |  |  |         self.seed = get_seed(seed) | 
					
						
							|  |  |  |         if secure: | 
					
						
							|  |  |  |             self.secure() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.random.seed(self.seed) | 
					
						
							|  |  |  |         self.seed_name = name if name else str(self.seed) | 
					
						
							|  |  |  |         self.slot_seeds = {player: random.Random(self.random.getrandbits(64)) for player in | 
					
						
							|  |  |  |                            range(1, self.players + 1)} | 
					
						
							| 
									
										
										
										
											2021-06-11 18:02:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-18 09:54:41 -07:00
										 |  |  |     def set_options(self, args: Namespace) -> None: | 
					
						
							| 
									
										
										
										
											2022-02-17 07:07:34 +01:00
										 |  |  |         for option_key in Options.common_options: | 
					
						
							|  |  |  |             setattr(self, option_key, getattr(args, option_key, {})) | 
					
						
							|  |  |  |         for option_key in Options.per_game_common_options: | 
					
						
							|  |  |  |             setattr(self, option_key, getattr(args, option_key, {})) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-11 18:02:48 +02:00
										 |  |  |         for player in self.player_ids: | 
					
						
							| 
									
										
										
										
											2021-04-24 01:16:49 +02:00
										 |  |  |             self.custom_data[player] = {} | 
					
						
							| 
									
										
										
										
											2021-07-04 16:18:21 +02:00
										 |  |  |             world_type = AutoWorld.AutoWorldRegister.world_types[self.game[player]] | 
					
						
							| 
									
										
										
										
											2022-08-15 16:46:59 -05:00
										 |  |  |             for option_key in world_type.option_definitions: | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  |                 setattr(self, option_key, getattr(args, option_key, {})) | 
					
						
							| 
									
										
										
										
											2022-02-17 07:07:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-04 16:18:21 +02:00
										 |  |  |             self.worlds[player] = world_type(self, player) | 
					
						
							| 
									
										
										
										
											2021-03-21 00:47:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-17 07:07:34 +01:00
										 |  |  |     def set_item_links(self): | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |         item_links = {} | 
					
						
							| 
									
										
										
										
											2022-12-07 06:37:47 +01:00
										 |  |  |         replacement_prio = [False, True, None] | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |         for player in self.player_ids: | 
					
						
							|  |  |  |             for item_link in self.item_links[player].value: | 
					
						
							|  |  |  |                 if item_link["name"] in item_links: | 
					
						
							| 
									
										
										
										
											2022-04-06 10:13:57 -07:00
										 |  |  |                     if item_links[item_link["name"]]["game"] != self.game[player]: | 
					
						
							| 
									
										
										
										
											2022-04-03 19:09:05 +02:00
										 |  |  |                         raise Exception(f"Cannot ItemLink across games. Link: {item_link['name']}") | 
					
						
							| 
									
										
										
										
											2022-12-07 06:37:47 +01:00
										 |  |  |                     current_link = item_links[item_link["name"]] | 
					
						
							|  |  |  |                     current_link["players"][player] = item_link["replacement_item"] | 
					
						
							|  |  |  |                     current_link["item_pool"] &= set(item_link["item_pool"]) | 
					
						
							|  |  |  |                     current_link["exclude"] |= set(item_link.get("exclude", [])) | 
					
						
							|  |  |  |                     current_link["local_items"] &= set(item_link.get("local_items", [])) | 
					
						
							|  |  |  |                     current_link["non_local_items"] &= set(item_link.get("non_local_items", [])) | 
					
						
							|  |  |  |                     current_link["link_replacement"] = min(current_link["link_replacement"], | 
					
						
							|  |  |  |                                                            replacement_prio.index(item_link["link_replacement"])) | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |                 else: | 
					
						
							|  |  |  |                     if item_link["name"] in self.player_name.values(): | 
					
						
							| 
									
										
										
										
											2022-12-07 06:37:47 +01:00
										 |  |  |                         raise Exception(f"Cannot name a ItemLink group the same as a player ({item_link['name']}) " | 
					
						
							|  |  |  |                                         f"({self.get_player_name(player)}).") | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |                     item_links[item_link["name"]] = { | 
					
						
							|  |  |  |                         "players": {player: item_link["replacement_item"]}, | 
					
						
							|  |  |  |                         "item_pool": set(item_link["item_pool"]), | 
					
						
							| 
									
										
										
										
											2022-05-11 16:37:18 -07:00
										 |  |  |                         "exclude": set(item_link.get("exclude", [])), | 
					
						
							| 
									
										
										
										
											2022-05-15 07:41:11 -07:00
										 |  |  |                         "game": self.game[player], | 
					
						
							|  |  |  |                         "local_items": set(item_link.get("local_items", [])), | 
					
						
							| 
									
										
										
										
											2022-12-07 06:37:47 +01:00
										 |  |  |                         "non_local_items": set(item_link.get("non_local_items", [])), | 
					
						
							|  |  |  |                         "link_replacement": replacement_prio.index(item_link["link_replacement"]), | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |                     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for name, item_link in item_links.items(): | 
					
						
							|  |  |  |             current_item_name_groups = AutoWorld.AutoWorldRegister.world_types[item_link["game"]].item_name_groups | 
					
						
							|  |  |  |             pool = set() | 
					
						
							| 
									
										
										
										
											2022-05-15 07:41:11 -07:00
										 |  |  |             local_items = set() | 
					
						
							|  |  |  |             non_local_items = set() | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |             for item in item_link["item_pool"]: | 
					
						
							|  |  |  |                 pool |= current_item_name_groups.get(item, {item}) | 
					
						
							| 
									
										
										
										
											2022-05-11 16:37:18 -07:00
										 |  |  |             for item in item_link["exclude"]: | 
					
						
							|  |  |  |                 pool -= current_item_name_groups.get(item, {item}) | 
					
						
							| 
									
										
										
										
											2022-05-15 07:41:11 -07:00
										 |  |  |             for item in item_link["local_items"]: | 
					
						
							|  |  |  |                 local_items |= current_item_name_groups.get(item, {item}) | 
					
						
							|  |  |  |             for item in item_link["non_local_items"]: | 
					
						
							|  |  |  |                 non_local_items |= current_item_name_groups.get(item, {item}) | 
					
						
							|  |  |  |             local_items &= pool | 
					
						
							|  |  |  |             non_local_items &= pool | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |             item_link["item_pool"] = pool | 
					
						
							| 
									
										
										
										
											2022-05-15 07:41:11 -07:00
										 |  |  |             item_link["local_items"] = local_items | 
					
						
							|  |  |  |             item_link["non_local_items"] = non_local_items | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for group_name, item_link in item_links.items(): | 
					
						
							|  |  |  |             game = item_link["game"] | 
					
						
							|  |  |  |             group_id, group = self.add_group(group_name, game, set(item_link["players"])) | 
					
						
							| 
									
										
										
										
											2022-12-07 06:37:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |             group["item_pool"] = item_link["item_pool"] | 
					
						
							|  |  |  |             group["replacement_items"] = item_link["players"] | 
					
						
							| 
									
										
										
										
											2022-05-15 07:41:11 -07:00
										 |  |  |             group["local_items"] = item_link["local_items"] | 
					
						
							|  |  |  |             group["non_local_items"] = item_link["non_local_items"] | 
					
						
							| 
									
										
										
										
											2022-12-07 06:37:47 +01:00
										 |  |  |             group["link_replacement"] = replacement_prio[item_link["link_replacement"]] | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  |     # intended for unittests | 
					
						
							|  |  |  |     def set_default_common_options(self): | 
					
						
							|  |  |  |         for option_key, option in Options.common_options.items(): | 
					
						
							|  |  |  |             setattr(self, option_key, {player_id: option(option.default) for player_id in self.player_ids}) | 
					
						
							|  |  |  |         for option_key, option in Options.per_game_common_options.items(): | 
					
						
							|  |  |  |             setattr(self, option_key, {player_id: option(option.default) for player_id in self.player_ids}) | 
					
						
							| 
									
										
										
										
											2022-02-17 07:07:34 +01:00
										 |  |  |         self.state = CollectionState(self) | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-14 07:01:51 +02:00
										 |  |  |     def secure(self): | 
					
						
							|  |  |  |         self.random = secrets.SystemRandom() | 
					
						
							| 
									
										
										
										
											2021-07-20 14:21:27 -04:00
										 |  |  |         self.is_race = True | 
					
						
							| 
									
										
										
										
											2020-07-14 07:01:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-04 16:44:27 +02:00
										 |  |  |     @functools.cached_property | 
					
						
							| 
									
										
										
										
											2022-11-20 20:50:32 +01:00
										 |  |  |     def player_ids(self) -> Tuple[int, ...]: | 
					
						
							| 
									
										
										
										
											2021-07-04 22:21:53 +02:00
										 |  |  |         return tuple(range(1, self.players + 1)) | 
					
						
							| 
									
										
										
										
											2020-06-19 03:01:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-21 18:08:15 +02:00
										 |  |  |     @functools.lru_cache() | 
					
						
							| 
									
										
										
										
											2022-11-20 20:50:32 +01:00
										 |  |  |     def get_game_players(self, game_name: str) -> Tuple[int, ...]: | 
					
						
							| 
									
										
										
										
											2021-07-21 18:08:15 +02:00
										 |  |  |         return tuple(player for player in self.player_ids if self.game[player] == game_name) | 
					
						
							| 
									
										
											  
											
												Minecraft Randomizer
Squash merge, original Commits:
* Minecraft locations, items, and generation without logic
* added id lookup for minecraft
* typing import fix in minecraft/Items.py
* fix 2
* implementing Minecraft options and hard/postgame advancement exclusion
* first logic pass (75/80)
* logic pass 2 and proper completion conditions
* added insane difficulty pool, modified method of excluding item pools for easier extension
* bump network_data_package version
* minecraft testing framework
* switch Ancient Debris to Netherite Scrap to avoid advancement triggering on receiving that item
* Testing now functions, split tests up by advancement pane, added some story tests
* Newer testing framework: every advancement gets its own function, for ease of testing
* fixed logic for The End... Again...
* changed option names to "include_hard_advancements" etc.
* village/pillager-related advancements now require can_adventure: weapon + food
* a few minecraft tests
* rename "Flint & Steel" to "Flint and Steel" for parity with in-game name
* additional MC tests
* more tests, mostly nether-related tests
* more tests, removed anvil path for Two Birds One Arrow
* include Minecraft slot data, and a world seed for each Minecraft player slot
* Added new items: ender pearls, lapis, porkchops
* All remaining Minecraft tests
* formatting of Minecraft tests and logic for better readability
* require Wither kill for Monsters Hunted
* properly removed 8 Emeralds item from item pool
* enchanting required for wither; fishing rod required for water breathing; water breathing required for elder guardian kill
* Added 12 new advancements (ported from old achievement system)
* renamed "On a Rail" for consistency with modern advancements
* tests for the new advancements
* moved slot_data generation for minecraft into worlds/minecraft/__init__.py, added logic_version to slot_data
* output minecraft options in the spoiler log
* modified advancement goal values for new advancements
* make non-native Minecraft items appear as Shovel in ALttP, and unknown-game items as Power Stars
* fixed glowstone block logic for Not Quite Nine Lives
* setup for shuffling MC structures: building ER world and shuffling regions/entrances
* ensured Nether Fortresses can't be placed in the End
* finished logic for structure randomization
* fixed nonnative items always showing up as Hammers in ALttP shops
* output minecraft structure info in the spoiler
* generate .apmc file for communication with MC client
* fixed structure rando always using the same seed
* move stuff to worlds/minecraft/Regions.py
* make output apmc file have consistent name with other files
* added minecraft bottle macro; fixed tests imports
* generalizing MC region generation
* restructured structure shuffling in preparation for structure plando
* only output structure rando info in spoiler if they are shuffled
* Force structure rando to always be off, for the stable release
* added Minecraft options to player settings
* formally added combat_difficulty as an option
* Added Ender Dragon into playthrough, cleaned up goal map
* Added new difficulties: Easy, Normal, Hard combat
* moved .apmc generation time to prevent outputs on failed generation
* updated tests for new combat logic
* Fixed bug causing generation to fail; removed Nether Fortress event since it should no longer be needed with the fix
* moved all MC-specific functions into gen_minecraft
* renamed "logic_version" to "client_version"
* bug fixes
properly flagged event locations/items with id None
moved generation back to Main.py to fix mysterious generation failures
* moved link_minecraft_regions into minecraft init, left create_regions in Main for caching
* added seed_name, player_name, client_version to apmc file
* reenabled structure shuffle
* added entrance tests for minecraft
Co-authored-by: achuang <alexander.w.chuang@gmail.com>
											
										 
											2021-05-08 07:38:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-30 16:31:56 +02:00
										 |  |  |     @functools.lru_cache() | 
					
						
							|  |  |  |     def get_game_worlds(self, game_name: str): | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  |         return tuple(world for player, world in self.worlds.items() if | 
					
						
							|  |  |  |                      player not in self.groups and self.game[player] == game_name) | 
					
						
							| 
									
										
										
										
											2021-08-30 16:31:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def get_name_string_for_object(self, obj) -> str: | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  |         return obj.name if self.players == 1 else f'{obj.name} ({self.get_player_name(obj.player)})' | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  |     def get_player_name(self, player: int) -> str: | 
					
						
							|  |  |  |         return self.player_name[player] | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-02 22:47:42 -05:00
										 |  |  |     def get_file_safe_player_name(self, player: int) -> str: | 
					
						
							|  |  |  |         return ''.join(c for c in self.get_player_name(player) if c not in '<>:"/\\|?*') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-02 07:53:18 -07:00
										 |  |  |     def get_out_file_name_base(self, player: int) -> str: | 
					
						
							|  |  |  |         """ the base name (without file extension) for each player's output file for a seed """ | 
					
						
							| 
									
										
										
										
											2022-11-20 13:39:52 -06:00
										 |  |  |         return f"AP_{self.seed_name}_P{player}_{self.get_file_safe_player_name(player).replace(' ', '_')}" | 
					
						
							| 
									
										
										
										
											2022-10-02 07:53:18 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-14 19:19:08 +01:00
										 |  |  |     def initialize_regions(self, regions=None): | 
					
						
							|  |  |  |         for region in regions if regions else self.regions: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             region.multiworld = self | 
					
						
							| 
									
										
										
										
											2019-12-14 19:19:08 +01:00
										 |  |  |             self._region_cache[region.player][region.name] = region | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-20 02:30:55 +01:00
										 |  |  |     @functools.cached_property | 
					
						
							|  |  |  |     def world_name_lookup(self): | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  |         return {self.player_name[player_id]: player_id for player_id in self.player_ids} | 
					
						
							| 
									
										
										
										
											2021-02-20 02:30:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-08 15:02:37 +02:00
										 |  |  |     def _recache(self): | 
					
						
							|  |  |  |         """Rebuild world cache""" | 
					
						
							| 
									
										
										
										
											2022-02-22 09:49:01 +01:00
										 |  |  |         self._cached_locations = None | 
					
						
							| 
									
										
										
										
											2020-09-08 15:02:37 +02:00
										 |  |  |         for region in self.regions: | 
					
						
							|  |  |  |             player = region.player | 
					
						
							|  |  |  |             self._region_cache[player][region.name] = region | 
					
						
							|  |  |  |             for exit in region.exits: | 
					
						
							|  |  |  |                 self._entrance_cache[exit.name, player] = exit | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for r_location in region.locations: | 
					
						
							|  |  |  |                 self._location_cache[r_location.name, player] = r_location | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-14 19:19:08 +01:00
										 |  |  |     def get_regions(self, player=None): | 
					
						
							|  |  |  |         return self.regions if player is None else self._region_cache[player].values() | 
					
						
							| 
									
										
										
										
											2017-10-28 18:34:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:17:36 +01:00
										 |  |  |     def get_region(self, regionname: str, player: int) -> Region: | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2019-12-14 19:19:08 +01:00
										 |  |  |             return self._region_cache[player][regionname] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         except KeyError: | 
					
						
							| 
									
										
										
										
											2020-09-08 15:02:37 +02:00
										 |  |  |             self._recache() | 
					
						
							|  |  |  |             return self._region_cache[player][regionname] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:17:36 +01:00
										 |  |  |     def get_entrance(self, entrance: str, player: int) -> Entrance: | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2020-09-08 15:02:37 +02:00
										 |  |  |             return self._entrance_cache[entrance, player] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         except KeyError: | 
					
						
							| 
									
										
										
										
											2020-09-08 15:02:37 +02:00
										 |  |  |             self._recache() | 
					
						
							|  |  |  |             return self._entrance_cache[entrance, player] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:17:36 +01:00
										 |  |  |     def get_location(self, location: str, player: int) -> Location: | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2020-09-08 15:02:37 +02:00
										 |  |  |             return self._location_cache[location, player] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         except KeyError: | 
					
						
							| 
									
										
										
										
											2020-09-08 15:02:37 +02:00
										 |  |  |             self._recache() | 
					
						
							|  |  |  |             return self._location_cache[location, player] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:17:36 +01:00
										 |  |  |     def get_dungeon(self, dungeonname: str, player: int) -> Dungeon: | 
					
						
							| 
									
										
										
										
											2021-08-29 16:02:28 +02:00
										 |  |  |         try: | 
					
						
							|  |  |  |             return self.dungeons[dungeonname, player] | 
					
						
							|  |  |  |         except KeyError as e: | 
					
						
							|  |  |  |             raise KeyError('No such dungeon %s for player %d' % (dungeonname, player)) from e | 
					
						
							| 
									
										
										
										
											2020-04-10 20:54:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-01 14:01:54 -05:00
										 |  |  |     def get_all_state(self, use_cache: bool) -> CollectionState: | 
					
						
							| 
									
										
										
										
											2021-08-31 19:19:26 -05:00
										 |  |  |         cached = getattr(self, "_all_state", None) | 
					
						
							| 
									
										
										
										
											2021-09-01 14:01:54 -05:00
										 |  |  |         if use_cache and cached: | 
					
						
							| 
									
										
										
										
											2021-08-09 06:33:26 +02:00
										 |  |  |             return cached.copy() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-17 14:40:37 +02:00
										 |  |  |         ret = CollectionState(self) | 
					
						
							| 
									
										
										
										
											2017-11-04 14:23:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-17 14:40:37 +02:00
										 |  |  |         for item in self.itempool: | 
					
						
							| 
									
										
										
										
											2021-07-04 15:47:11 +02:00
										 |  |  |             self.worlds[item.player].collect(ret, item) | 
					
						
							| 
									
										
										
										
											2022-02-13 23:02:18 +01:00
										 |  |  |         for player in self.player_ids: | 
					
						
							|  |  |  |             subworld = self.worlds[player] | 
					
						
							|  |  |  |             for item in subworld.get_pre_fill_items(): | 
					
						
							| 
									
										
										
										
											2021-08-31 19:19:26 -05:00
										 |  |  |                 subworld.collect(ret, item) | 
					
						
							| 
									
										
										
										
											2017-07-17 23:14:31 +02:00
										 |  |  |         ret.sweep_for_events() | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-01 14:01:54 -05:00
										 |  |  |         if use_cache: | 
					
						
							| 
									
										
										
										
											2021-09-01 13:20:43 -05:00
										 |  |  |             self._all_state = ret | 
					
						
							| 
									
										
										
										
											2017-06-17 14:40:37 +02:00
										 |  |  |         return ret | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-31 14:23:01 -07:00
										 |  |  |     def get_items(self) -> List[Item]: | 
					
						
							| 
									
										
										
										
											2018-01-02 20:01:16 -05:00
										 |  |  |         return [loc.item for loc in self.get_filled_locations()] + self.itempool | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-27 22:57:54 +01:00
										 |  |  |     def find_item_locations(self, item, player: int) -> List[Location]: | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |         return [location for location in self.get_locations() if | 
					
						
							| 
									
										
										
										
											2021-11-27 22:57:54 +01:00
										 |  |  |                 location.item and location.item.name == item and location.item.player == player] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-14 00:27:06 +01:00
										 |  |  |     def find_item(self, item, player: int) -> Location: | 
					
						
							|  |  |  |         return next(location for location in self.get_locations() if | 
					
						
							|  |  |  |                     location.item and location.item.name == item and location.item.player == player) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-27 22:57:54 +01:00
										 |  |  |     def find_items_in_locations(self, items: Set[str], player: int) -> List[Location]: | 
					
						
							|  |  |  |         return [location for location in self.get_locations() if | 
					
						
							|  |  |  |                 location.item and location.item.name in items and location.item.player == player] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 13:54:47 +02:00
										 |  |  |     def create_item(self, item_name: str, player: int) -> Item: | 
					
						
							|  |  |  |         return self.worlds[player].create_item(item_name) | 
					
						
							| 
									
										
										
										
											2021-03-14 00:27:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def push_precollected(self, item: Item): | 
					
						
							| 
									
										
										
										
											2021-10-10 16:50:01 +02:00
										 |  |  |         self.precollected_items[item.player].append(item) | 
					
						
							| 
									
										
										
										
											2019-08-10 15:30:14 -04:00
										 |  |  |         self.state.collect(item, True) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def push_item(self, location: Location, item: Item, collect: bool = True): | 
					
						
							| 
									
										
										
										
											2022-07-14 09:46:03 +02:00
										 |  |  |         assert location.can_fill(self.state, item, False), f"Cannot place {item} into {location}." | 
					
						
							|  |  |  |         location.item = item | 
					
						
							|  |  |  |         item.location = location | 
					
						
							|  |  |  |         if collect: | 
					
						
							|  |  |  |             self.state.collect(item, location.event, location) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         logging.debug('Placed %s at %s', item, location) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |     def get_entrances(self) -> List[Entrance]: | 
					
						
							| 
									
										
										
										
											2019-01-20 01:01:02 -06:00
										 |  |  |         if self._cached_entrances is None: | 
					
						
							| 
									
										
										
										
											2020-08-27 04:05:11 +02:00
										 |  |  |             self._cached_entrances = [entrance for region in self.regions for entrance in region.entrances] | 
					
						
							| 
									
										
										
										
											2019-01-20 01:01:02 -06:00
										 |  |  |         return self._cached_entrances | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def clear_entrance_cache(self): | 
					
						
							|  |  |  |         self._cached_entrances = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-30 04:58:19 +02:00
										 |  |  |     def register_indirect_condition(self, region: Region, entrance: Entrance): | 
					
						
							|  |  |  |         """Report that access to this Region can result in unlocking this Entrance,
 | 
					
						
							|  |  |  |         state.can_reach(Region) in the Entrance's traversal condition, as opposed to pure transition logic.""" | 
					
						
							|  |  |  |         self.indirect_connections.setdefault(region, set()).add(entrance) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-20 20:50:32 +01:00
										 |  |  |     def get_locations(self, player: Optional[int] = None) -> List[Location]: | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         if self._cached_locations is None: | 
					
						
							| 
									
										
										
										
											2020-08-27 04:05:11 +02:00
										 |  |  |             self._cached_locations = [location for region in self.regions for location in region.locations] | 
					
						
							| 
									
										
										
										
											2022-11-20 20:50:32 +01:00
										 |  |  |         if player is not None: | 
					
						
							|  |  |  |             return [location for location in self._cached_locations if location.player == player] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         return self._cached_locations | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-22 23:18:40 -04:00
										 |  |  |     def clear_location_cache(self): | 
					
						
							|  |  |  |         self._cached_locations = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  |     def get_unfilled_locations(self, player: Optional[int] = None) -> List[Location]: | 
					
						
							| 
									
										
										
										
											2022-11-20 20:50:32 +01:00
										 |  |  |         return [location for location in self.get_locations(player) if location.item is None] | 
					
						
							| 
									
										
										
										
											2020-10-07 19:51:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  |     def get_filled_locations(self, player: Optional[int] = None) -> List[Location]: | 
					
						
							| 
									
										
										
										
											2022-11-20 20:50:32 +01:00
										 |  |  |         return [location for location in self.get_locations(player) if location.item is not None] | 
					
						
							| 
									
										
										
										
											2017-06-17 14:40:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  |     def get_reachable_locations(self, state: Optional[CollectionState] = None, player: Optional[int] = None) -> List[Location]: | 
					
						
							| 
									
										
										
										
											2022-11-20 20:50:32 +01:00
										 |  |  |         state: CollectionState = state if state else self.state | 
					
						
							|  |  |  |         return [location for location in self.get_locations(player) if location.can_reach(state)] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |     def get_placeable_locations(self, state=None, player=None) -> List[Location]: | 
					
						
							| 
									
										
										
										
											2022-11-20 20:50:32 +01:00
										 |  |  |         state: CollectionState = state if state else self.state | 
					
						
							|  |  |  |         return [location for location in self.get_locations(player) if location.item is None and location.can_reach(state)] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-20 20:50:32 +01:00
										 |  |  |     def get_unfilled_locations_for_players(self, location_names: List[str], players: Iterable[int]): | 
					
						
							| 
									
										
										
										
											2021-01-04 15:14:20 +01:00
										 |  |  |         for player in players: | 
					
						
							| 
									
										
										
										
											2022-11-20 20:50:32 +01:00
										 |  |  |             if not location_names: | 
					
						
							|  |  |  |                 location_names = [location.name for location in self.get_unfilled_locations(player)] | 
					
						
							|  |  |  |             for location_name in location_names: | 
					
						
							| 
									
										
										
										
											2022-01-20 13:34:17 -05:00
										 |  |  |                 location = self._location_cache.get((location_name, player), None) | 
					
						
							|  |  |  |                 if location is not None and location.item is None: | 
					
						
							|  |  |  |                     yield location | 
					
						
							| 
									
										
										
										
											2021-01-04 15:14:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  |     def unlocks_new_location(self, item: Item) -> bool: | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         temp_state = self.state.copy() | 
					
						
							| 
									
										
										
										
											2017-06-17 14:40:37 +02:00
										 |  |  |         temp_state.collect(item, True) | 
					
						
							| 
									
										
										
										
											2017-05-26 09:55:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for location in self.get_unfilled_locations(): | 
					
						
							|  |  |  |             if temp_state.can_reach(location) and not self.state.can_reach(location): | 
					
						
							|  |  |  |                 return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return False | 
					
						
							| 
									
										
										
										
											2017-11-18 20:43:37 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  |     def has_beaten_game(self, state: CollectionState, player: Optional[int] = None) -> bool: | 
					
						
							| 
									
										
										
										
											2019-07-09 22:18:24 -04:00
										 |  |  |         if player: | 
					
						
							| 
									
										
										
										
											2021-02-22 11:18:53 +01:00
										 |  |  |             return self.completion_condition[player](state) | 
					
						
							| 
									
										
										
										
											2019-07-09 22:18:24 -04:00
										 |  |  |         else: | 
					
						
							|  |  |  |             return all((self.has_beaten_game(state, p) for p in range(1, self.players + 1))) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |     def can_beat_game(self, starting_state: Optional[CollectionState] = None): | 
					
						
							| 
									
										
										
										
											2017-06-23 22:15:29 +02:00
										 |  |  |         if starting_state: | 
					
						
							| 
									
										
										
										
											2020-03-07 23:20:11 +01:00
										 |  |  |             if self.has_beaten_game(starting_state): | 
					
						
							|  |  |  |                 return True | 
					
						
							| 
									
										
										
										
											2017-06-23 22:15:29 +02:00
										 |  |  |             state = starting_state.copy() | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2020-03-07 23:20:11 +01:00
										 |  |  |             if self.has_beaten_game(self.state): | 
					
						
							|  |  |  |                 return True | 
					
						
							| 
									
										
										
										
											2017-06-23 22:15:29 +02:00
										 |  |  |             state = CollectionState(self) | 
					
						
							| 
									
										
										
										
											2021-04-29 09:54:49 +02:00
										 |  |  |         prog_locations = {location for location in self.get_locations() if location.item | 
					
						
							|  |  |  |                           and location.item.advancement and location not in state.locations_checked} | 
					
						
							| 
									
										
										
										
											2018-01-01 15:55:13 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-16 21:23:47 +02:00
										 |  |  |         while prog_locations: | 
					
						
							| 
									
										
										
										
											2021-04-29 09:54:49 +02:00
										 |  |  |             sphere = set() | 
					
						
							| 
									
										
										
										
											2021-03-07 22:05:07 +01:00
										 |  |  |             # build up spheres of collection radius. | 
					
						
							|  |  |  |             # Everything in each sphere is independent from each other in dependencies and only depends on lower spheres | 
					
						
							| 
									
										
										
										
											2017-05-16 21:23:47 +02:00
										 |  |  |             for location in prog_locations: | 
					
						
							| 
									
										
										
										
											2019-07-11 00:12:09 -04:00
										 |  |  |                 if location.can_reach(state): | 
					
						
							| 
									
										
										
										
											2021-04-29 09:54:49 +02:00
										 |  |  |                     sphere.add(location) | 
					
						
							| 
									
										
										
										
											2017-05-16 21:23:47 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if not sphere: | 
					
						
							| 
									
										
										
										
											2019-07-11 00:12:09 -04:00
										 |  |  |                 # ran out of places and did not finish yet, quit | 
					
						
							| 
									
										
										
										
											2017-05-16 21:23:47 +02:00
										 |  |  |                 return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for location in sphere: | 
					
						
							| 
									
										
										
										
											2018-01-01 15:55:13 -05:00
										 |  |  |                 state.collect(location.item, True, location) | 
					
						
							| 
									
										
										
										
											2021-04-29 09:54:49 +02:00
										 |  |  |             prog_locations -= sphere | 
					
						
							| 
									
										
										
										
											2017-05-16 21:23:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-11 00:12:09 -04:00
										 |  |  |             if self.has_beaten_game(state): | 
					
						
							|  |  |  |                 return True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-16 21:23:47 +02:00
										 |  |  |         return False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-17 22:58:52 +01:00
										 |  |  |     def get_spheres(self): | 
					
						
							|  |  |  |         state = CollectionState(self) | 
					
						
							| 
									
										
										
										
											2021-02-27 18:58:17 +01:00
										 |  |  |         locations = set(self.get_filled_locations()) | 
					
						
							| 
									
										
										
										
											2021-01-17 22:58:52 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         while locations: | 
					
						
							|  |  |  |             sphere = set() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for location in locations: | 
					
						
							|  |  |  |                 if location.can_reach(state): | 
					
						
							|  |  |  |                     sphere.add(location) | 
					
						
							| 
									
										
										
										
											2021-02-27 18:58:17 +01:00
										 |  |  |             yield sphere | 
					
						
							| 
									
										
										
										
											2021-01-17 22:58:52 +01:00
										 |  |  |             if not sphere: | 
					
						
							|  |  |  |                 if locations: | 
					
						
							|  |  |  |                     yield locations  # unreachable locations | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for location in sphere: | 
					
						
							|  |  |  |                 state.collect(location.item, True, location) | 
					
						
							|  |  |  |             locations -= sphere | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-13 14:27:17 +01:00
										 |  |  |     def fulfills_accessibility(self, state: Optional[CollectionState] = None): | 
					
						
							|  |  |  |         """Check if accessibility rules are fulfilled with current or supplied state.""" | 
					
						
							|  |  |  |         if not state: | 
					
						
							|  |  |  |             state = CollectionState(self) | 
					
						
							| 
									
										
										
										
											2022-10-30 15:13:53 +01:00
										 |  |  |         players: Dict[str, Set[int]] = { | 
					
						
							|  |  |  |             "minimal": set(), | 
					
						
							|  |  |  |             "items": set(), | 
					
						
							|  |  |  |             "locations": set() | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-01-13 14:27:17 +01:00
										 |  |  |         for player, access in self.accessibility.items(): | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  |             players[access.current_key].add(player) | 
					
						
							| 
									
										
										
										
											2021-01-11 19:56:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         beatable_fulfilled = False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-29 13:59:57 -04:00
										 |  |  |         def location_condition(location: Location): | 
					
						
							| 
									
										
										
										
											2021-01-13 14:27:17 +01:00
										 |  |  |             """Determine if this location has to be accessible, location is already filtered by location_relevant""" | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  |             if location.player in players["minimal"]: | 
					
						
							| 
									
										
										
										
											2021-01-13 14:27:17 +01:00
										 |  |  |                 return False | 
					
						
							|  |  |  |             return True | 
					
						
							| 
									
										
										
										
											2021-01-11 19:56:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |         def location_relevant(location: Location): | 
					
						
							| 
									
										
										
										
											2021-01-13 14:27:17 +01:00
										 |  |  |             """Determine if this location is relevant to sweep.""" | 
					
						
							| 
									
										
										
										
											2022-03-29 02:03:57 +02:00
										 |  |  |             if location.progress_type != LocationProgressType.EXCLUDED \ | 
					
						
							|  |  |  |                 and (location.player in players["locations"] or location.event | 
					
						
							|  |  |  |                      or (location.item and location.item.advancement)): | 
					
						
							| 
									
										
										
										
											2021-01-13 14:27:17 +01:00
										 |  |  |                 return True | 
					
						
							| 
									
										
										
										
											2021-01-11 19:56:18 +01:00
										 |  |  |             return False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-30 15:13:53 +01:00
										 |  |  |         def all_done() -> bool: | 
					
						
							| 
									
										
										
										
											2021-01-11 19:56:18 +01:00
										 |  |  |             """Check if all access rules are fulfilled""" | 
					
						
							| 
									
										
										
										
											2022-10-30 15:13:53 +01:00
										 |  |  |             if not beatable_fulfilled: | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  |             if any(location_condition(location) for location in locations): | 
					
						
							|  |  |  |                 return False  # still locations required to be collected | 
					
						
							|  |  |  |             return True | 
					
						
							| 
									
										
										
										
											2021-01-11 19:56:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-30 15:13:53 +01:00
										 |  |  |         locations = [location for location in self.get_locations() if location_relevant(location)] | 
					
						
							| 
									
										
										
										
											2021-01-13 14:27:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-11 19:56:18 +01:00
										 |  |  |         while locations: | 
					
						
							| 
									
										
										
										
											2022-10-30 15:13:53 +01:00
										 |  |  |             sphere: List[Location] = [] | 
					
						
							| 
									
										
										
										
											2022-10-31 20:29:22 +01:00
										 |  |  |             for n in range(len(locations) - 1, -1, -1): | 
					
						
							|  |  |  |                 if locations[n].can_reach(state): | 
					
						
							|  |  |  |                     sphere.append(locations.pop(n)) | 
					
						
							| 
									
										
										
										
											2021-01-11 19:56:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if not sphere: | 
					
						
							|  |  |  |                 # ran out of places and did not finish yet, quit | 
					
						
							| 
									
										
										
										
											2021-02-26 21:03:16 +01:00
										 |  |  |                 logging.warning(f"Could not access required locations for accessibility check." | 
					
						
							|  |  |  |                                 f" Missing: {locations}") | 
					
						
							| 
									
										
										
										
											2021-01-11 19:56:18 +01:00
										 |  |  |                 return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for location in sphere: | 
					
						
							| 
									
										
										
										
											2022-10-30 15:13:53 +01:00
										 |  |  |                 if location.item: | 
					
						
							|  |  |  |                     state.collect(location.item, True, location) | 
					
						
							| 
									
										
										
										
											2021-01-11 19:56:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if self.has_beaten_game(state): | 
					
						
							|  |  |  |                 beatable_fulfilled = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if all_done(): | 
					
						
							|  |  |  |                 return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return False | 
					
						
							| 
									
										
										
										
											2017-05-16 21:23:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-13 14:27:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-27 12:19:53 -07:00
										 |  |  | PathValue = Tuple[str, Optional["PathValue"]] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-17 07:07:34 +01:00
										 |  |  | class CollectionState(): | 
					
						
							| 
									
										
										
										
											2022-04-27 12:19:53 -07:00
										 |  |  |     prog_items: typing.Counter[Tuple[str, int]] | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |     multiworld: MultiWorld | 
					
						
							| 
									
										
										
										
											2022-04-27 12:19:53 -07:00
										 |  |  |     reachable_regions: Dict[int, Set[Region]] | 
					
						
							|  |  |  |     blocked_connections: Dict[int, Set[Entrance]] | 
					
						
							|  |  |  |     events: Set[Location] | 
					
						
							|  |  |  |     path: Dict[Union[Region, Entrance], PathValue] | 
					
						
							|  |  |  |     locations_checked: Set[Location] | 
					
						
							|  |  |  |     stale: Dict[int, bool] | 
					
						
							|  |  |  |     additional_init_functions: List[Callable[[CollectionState, MultiWorld], None]] = [] | 
					
						
							|  |  |  |     additional_copy_functions: List[Callable[[CollectionState, CollectionState], CollectionState]] = [] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-24 05:38:56 +02:00
										 |  |  |     def __init__(self, parent: MultiWorld): | 
					
						
							| 
									
										
										
										
											2020-03-07 23:35:55 +01:00
										 |  |  |         self.prog_items = Counter() | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         self.multiworld = parent | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |         self.reachable_regions = {player: set() for player in parent.get_all_ids()} | 
					
						
							|  |  |  |         self.blocked_connections = {player: set() for player in parent.get_all_ids()} | 
					
						
							| 
									
										
										
										
											2020-08-22 19:19:29 +02:00
										 |  |  |         self.events = set() | 
					
						
							| 
									
										
										
										
											2018-01-01 15:55:13 -05:00
										 |  |  |         self.path = {} | 
					
						
							|  |  |  |         self.locations_checked = set() | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |         self.stale = {player: True for player in parent.get_all_ids()} | 
					
						
							| 
									
										
										
										
											2022-02-19 17:43:16 +01:00
										 |  |  |         for function in self.additional_init_functions: | 
					
						
							|  |  |  |             function(self, parent) | 
					
						
							| 
									
										
										
										
											2021-10-10 16:50:01 +02:00
										 |  |  |         for items in parent.precollected_items.values(): | 
					
						
							|  |  |  |             for item in items: | 
					
						
							|  |  |  |                 self.collect(item, True) | 
					
						
							| 
									
										
										
										
											2018-01-01 15:55:13 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def update_reachable_regions(self, player: int): | 
					
						
							| 
									
										
										
										
											2019-07-11 00:18:30 -04:00
										 |  |  |         self.stale[player] = False | 
					
						
							|  |  |  |         rrp = self.reachable_regions[player] | 
					
						
							| 
									
										
										
										
											2020-05-10 19:27:13 +10:00
										 |  |  |         bc = self.blocked_connections[player] | 
					
						
							|  |  |  |         queue = deque(self.blocked_connections[player]) | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         start = self.multiworld.get_region('Menu', player) | 
					
						
							| 
									
										
										
										
											2020-05-10 19:27:13 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # init on first call - this can't be done on construction since the regions don't exist yet | 
					
						
							| 
									
										
										
										
											2022-09-30 04:58:19 +02:00
										 |  |  |         if start not in rrp: | 
					
						
							| 
									
										
										
										
											2020-05-10 19:27:13 +10:00
										 |  |  |             rrp.add(start) | 
					
						
							|  |  |  |             bc.update(start.exits) | 
					
						
							|  |  |  |             queue.extend(start.exits) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # run BFS on all connections, and keep track of those blocked by missing items | 
					
						
							| 
									
										
										
										
											2020-06-30 07:32:05 +02:00
										 |  |  |         while queue: | 
					
						
							|  |  |  |             connection = queue.popleft() | 
					
						
							|  |  |  |             new_region = connection.connected_region | 
					
						
							|  |  |  |             if new_region in rrp: | 
					
						
							|  |  |  |                 bc.remove(connection) | 
					
						
							|  |  |  |             elif connection.can_reach(self): | 
					
						
							| 
									
										
										
										
											2022-11-03 09:17:34 -05:00
										 |  |  |                 assert new_region, f"tried to search through an Entrance \"{connection}\" with no Region" | 
					
						
							| 
									
										
										
										
											2020-06-30 07:32:05 +02:00
										 |  |  |                 rrp.add(new_region) | 
					
						
							|  |  |  |                 bc.remove(connection) | 
					
						
							|  |  |  |                 bc.update(new_region.exits) | 
					
						
							|  |  |  |                 queue.extend(new_region.exits) | 
					
						
							|  |  |  |                 self.path[new_region] = (new_region.name, self.path.get(connection, None)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # Retry connections if the new region can unblock them | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 for new_entrance in self.multiworld.indirect_connections.get(new_region, set()): | 
					
						
							| 
									
										
										
										
											2020-06-30 07:32:05 +02:00
										 |  |  |                     if new_entrance in bc and new_entrance not in queue: | 
					
						
							|  |  |  |                         queue.append(new_entrance) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def copy(self) -> CollectionState: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         ret = CollectionState(self.multiworld) | 
					
						
							| 
									
										
										
										
											2019-07-13 18:17:16 -04:00
										 |  |  |         ret.prog_items = self.prog_items.copy() | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |         ret.reachable_regions = {player: copy.copy(self.reachable_regions[player]) for player in | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |                                  self.reachable_regions} | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |         ret.blocked_connections = {player: copy.copy(self.blocked_connections[player]) for player in | 
					
						
							| 
									
										
										
										
											2022-02-17 06:07:11 +01:00
										 |  |  |                                    self.blocked_connections} | 
					
						
							| 
									
										
										
										
											2017-06-17 14:40:37 +02:00
										 |  |  |         ret.events = copy.copy(self.events) | 
					
						
							| 
									
										
										
										
											2018-01-01 15:55:13 -05:00
										 |  |  |         ret.path = copy.copy(self.path) | 
					
						
							|  |  |  |         ret.locations_checked = copy.copy(self.locations_checked) | 
					
						
							| 
									
										
										
										
											2022-02-17 07:07:34 +01:00
										 |  |  |         for function in self.additional_copy_functions: | 
					
						
							|  |  |  |             ret = function(self, ret) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         return ret | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  |     def can_reach(self, | 
					
						
							|  |  |  |                   spot: Union[Location, Entrance, Region, str], | 
					
						
							|  |  |  |                   resolution_hint: Optional[str] = None, | 
					
						
							|  |  |  |                   player: Optional[int] = None) -> bool: | 
					
						
							| 
									
										
										
										
											2022-04-27 12:19:53 -07:00
										 |  |  |         if isinstance(spot, str): | 
					
						
							|  |  |  |             assert isinstance(player, int), "can_reach: player is required if spot is str" | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |             # try to resolve a name | 
					
						
							|  |  |  |             if resolution_hint == 'Location': | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 spot = self.multiworld.get_location(spot, player) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |             elif resolution_hint == 'Entrance': | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 spot = self.multiworld.get_entrance(spot, player) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 # default to Region | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 spot = self.multiworld.get_region(spot, player) | 
					
						
							| 
									
										
										
										
											2019-07-08 22:48:16 -04:00
										 |  |  |         return spot.can_reach(self) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-27 12:19:53 -07:00
										 |  |  |     def sweep_for_events(self, key_only: bool = False, locations: Optional[Iterable[Location]] = None) -> None: | 
					
						
							| 
									
										
										
										
											2019-12-13 22:37:52 +01:00
										 |  |  |         if locations is None: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             locations = self.multiworld.get_filled_locations() | 
					
						
							| 
									
										
										
										
											2022-10-12 23:46:07 -04:00
										 |  |  |         reachable_events = True | 
					
						
							| 
									
										
										
										
											2021-02-14 17:52:01 +01:00
										 |  |  |         # since the loop has a good chance to run more than once, only filter the events once | 
					
						
							| 
									
										
										
										
											2022-10-18 02:22:05 +02:00
										 |  |  |         locations = {location for location in locations if location.event and location not in self.events and | 
					
						
							| 
									
										
										
										
											2022-03-31 03:30:06 +02:00
										 |  |  |                      not key_only or getattr(location.item, "locked_dungeon_item", False)} | 
					
						
							| 
									
										
										
										
											2022-10-12 23:46:07 -04:00
										 |  |  |         while reachable_events: | 
					
						
							| 
									
										
										
										
											2022-03-31 03:30:06 +02:00
										 |  |  |             reachable_events = {location for location in locations if location.can_reach(self)} | 
					
						
							| 
									
										
										
										
											2022-10-12 23:46:07 -04:00
										 |  |  |             locations -= reachable_events | 
					
						
							|  |  |  |             for event in reachable_events: | 
					
						
							| 
									
										
										
										
											2020-08-22 19:19:29 +02:00
										 |  |  |                 self.events.add(event) | 
					
						
							| 
									
										
										
										
											2022-04-27 12:19:53 -07:00
										 |  |  |                 assert isinstance(event.item, Item), "tried to collect Event with no Item" | 
					
						
							| 
									
										
										
										
											2020-08-22 19:19:29 +02:00
										 |  |  |                 self.collect(event.item, True, event) | 
					
						
							| 
									
										
										
										
											2019-07-13 18:17:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-26 01:12:54 +01:00
										 |  |  |     def has(self, item: str, player: int, count: int = 1) -> bool: | 
					
						
							| 
									
										
										
										
											2020-03-07 23:35:55 +01:00
										 |  |  |         return self.prog_items[item, player] >= count | 
					
						
							| 
									
										
										
										
											2017-11-18 20:43:37 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-26 01:12:54 +01:00
										 |  |  |     def has_all(self, items: Set[str], player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2021-07-15 08:50:08 +02:00
										 |  |  |         return all(self.prog_items[item, player] for item in items) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-26 01:12:54 +01:00
										 |  |  |     def has_any(self, items: Set[str], player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2021-07-15 08:50:08 +02:00
										 |  |  |         return any(self.prog_items[item, player] for item in items) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-26 01:12:54 +01:00
										 |  |  |     def count(self, item: str, player: int) -> int: | 
					
						
							|  |  |  |         return self.prog_items[item, player] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-27 12:19:53 -07:00
										 |  |  |     def has_group(self, item_name_group: str, player: int, count: int = 1) -> bool: | 
					
						
							| 
									
										
										
										
											2021-07-21 09:45:15 +02:00
										 |  |  |         found: int = 0 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         for item_name in self.multiworld.worlds[player].item_name_groups[item_name_group]: | 
					
						
							| 
									
										
										
										
											2021-07-21 09:45:15 +02:00
										 |  |  |             found += self.prog_items[item_name, player] | 
					
						
							|  |  |  |             if found >= count: | 
					
						
							|  |  |  |                 return True | 
					
						
							|  |  |  |         return False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-27 12:19:53 -07:00
										 |  |  |     def count_group(self, item_name_group: str, player: int) -> int: | 
					
						
							| 
									
										
										
										
											2021-07-21 09:45:15 +02:00
										 |  |  |         found: int = 0 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         for item_name in self.multiworld.worlds[player].item_name_groups[item_name_group]: | 
					
						
							| 
									
										
										
										
											2021-07-21 09:45:15 +02:00
										 |  |  |             found += self.prog_items[item_name, player] | 
					
						
							|  |  |  |         return found | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def can_buy_unlimited(self, item: str, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2020-08-20 20:13:00 +02:00
										 |  |  |         return any(shop.region.player == player and shop.has_unlimited(item) and shop.region.can_reach(self) for | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                    shop in self.multiworld.shops) | 
					
						
							| 
									
										
										
										
											2018-02-17 18:38:54 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-23 21:38:21 +02:00
										 |  |  |     def can_buy(self, item: str, player: int) -> bool: | 
					
						
							|  |  |  |         return any(shop.region.player == player and shop.has(item) and shop.region.can_reach(self) for | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                    shop in self.multiworld.shops) | 
					
						
							| 
									
										
										
										
											2020-08-23 21:38:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-30 05:33:56 +01:00
										 |  |  |     def item_count(self, item: str, player: int) -> int: | 
					
						
							| 
									
										
										
										
											2020-03-07 23:35:55 +01:00
										 |  |  |         return self.prog_items[item, player] | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-26 07:18:53 -07:00
										 |  |  |     def has_triforce_pieces(self, count: int, player: int) -> bool: | 
					
						
							|  |  |  |         return self.item_count('Triforce Piece', player) + self.item_count('Power Star', player) >= count | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def has_crystals(self, count: int, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2020-08-14 00:34:41 +02:00
										 |  |  |         found: int = 0 | 
					
						
							| 
									
										
										
										
											2020-08-22 19:19:29 +02:00
										 |  |  |         for crystalnumber in range(1, 8): | 
					
						
							|  |  |  |             found += self.prog_items[f"Crystal {crystalnumber}", player] | 
					
						
							|  |  |  |             if found >= count: | 
					
						
							|  |  |  |                 return True | 
					
						
							| 
									
										
										
										
											2020-08-14 00:34:41 +02:00
										 |  |  |         return False | 
					
						
							| 
									
										
										
										
											2019-07-25 18:25:14 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-14 00:34:41 +02:00
										 |  |  |     def can_lift_rocks(self, player: int): | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         return self.has('Power Glove', player) or self.has('Titans Mitts', player) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def bottle_count(self, player: int) -> int: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         return min(self.multiworld.difficulty_requirements[player].progressive_bottle_limit, | 
					
						
							| 
									
										
										
										
											2021-07-21 09:45:15 +02:00
										 |  |  |                    self.count_group("Bottles", player)) | 
					
						
							| 
									
										
										
										
											2017-11-11 20:22:44 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def has_hearts(self, player: int, count: int) -> int: | 
					
						
							| 
									
										
										
										
											2018-09-16 12:55:49 -04:00
										 |  |  |         # Warning: This only considers items that are marked as advancement items | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         return self.heart_count(player) >= count | 
					
						
							| 
									
										
										
										
											2018-01-06 13:39:22 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def heart_count(self, player: int) -> int: | 
					
						
							| 
									
										
										
										
											2018-09-16 12:55:49 -04:00
										 |  |  |         # Warning: This only considers items that are marked as advancement items | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         diff = self.multiworld.difficulty_requirements[player] | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |         return min(self.item_count('Boss Heart Container', player), diff.boss_heart_container_limit) \ | 
					
						
							|  |  |  |                + self.item_count('Sanctuary Heart Container', player) \ | 
					
						
							|  |  |  |                + min(self.item_count('Piece of Heart', player), diff.heart_piece_limit) // 4 \ | 
					
						
							|  |  |  |                + 3  # starting hearts | 
					
						
							| 
									
										
										
										
											2018-01-06 13:39:22 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def can_lift_heavy_rocks(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         return self.has('Titans Mitts', player) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def can_extend_magic(self, player: int, smallmagic: int = 16, | 
					
						
							|  |  |  |                          fullrefill: bool = False):  # This reflects the total magic Link has, not the total extra he has. | 
					
						
							| 
									
										
										
										
											2018-01-06 21:07:46 -06:00
										 |  |  |         basemagic = 8 | 
					
						
							| 
									
										
										
										
											2020-03-14 10:31:28 +11:00
										 |  |  |         if self.has('Magic Upgrade (1/4)', player): | 
					
						
							| 
									
										
										
										
											2018-01-06 21:07:46 -06:00
										 |  |  |             basemagic = 32 | 
					
						
							| 
									
										
										
										
											2020-03-14 10:31:28 +11:00
										 |  |  |         elif self.has('Magic Upgrade (1/2)', player): | 
					
						
							| 
									
										
										
										
											2018-01-06 21:07:46 -06:00
										 |  |  |             basemagic = 16 | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         if self.can_buy_unlimited('Green Potion', player) or self.can_buy_unlimited('Blue Potion', player): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             if self.multiworld.item_functionality[player] == 'hard' and not fullrefill: | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |                 basemagic = basemagic + int(basemagic * 0.5 * self.bottle_count(player)) | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             elif self.multiworld.item_functionality[player] == 'expert' and not fullrefill: | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |                 basemagic = basemagic + int(basemagic * 0.25 * self.bottle_count(player)) | 
					
						
							| 
									
										
										
										
											2018-09-16 12:55:49 -04:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |                 basemagic = basemagic + basemagic * self.bottle_count(player) | 
					
						
							| 
									
										
										
										
											2018-02-17 18:38:54 -05:00
										 |  |  |         return basemagic >= smallmagic | 
					
						
							| 
									
										
										
										
											2018-01-02 00:39:53 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def can_kill_most_things(self, player: int, enemies: int = 5) -> bool: | 
					
						
							| 
									
										
										
										
											2020-04-20 19:17:10 +02:00
										 |  |  |         return (self.has_melee_weapon(player) | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |                 or self.has('Cane of Somaria', player) | 
					
						
							|  |  |  |                 or (self.has('Cane of Byrna', player) and (enemies < 6 or self.can_extend_magic(player))) | 
					
						
							|  |  |  |                 or self.can_shoot_arrows(player) | 
					
						
							|  |  |  |                 or self.has('Fire Rod', player) | 
					
						
							| 
									
										
										
										
											2020-03-15 21:59:06 +11:00
										 |  |  |                 or (self.has('Bombs (10)', player) and enemies < 6)) | 
					
						
							| 
									
										
										
										
											2018-01-02 00:39:53 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def can_shoot_arrows(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         if self.multiworld.retro_bow[player]: | 
					
						
							| 
									
										
										
										
											2020-08-23 21:38:21 +02:00
										 |  |  |             return (self.has('Bow', player) or self.has('Silver Bow', player)) and self.can_buy('Single Arrow', player) | 
					
						
							| 
									
										
										
										
											2020-06-30 09:51:11 +02:00
										 |  |  |         return self.has('Bow', player) or self.has('Silver Bow', player) | 
					
						
							| 
									
										
										
										
											2018-02-17 18:38:54 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def can_get_good_bee(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         cave = self.multiworld.get_region('Good Bee Cave', player) | 
					
						
							| 
									
										
										
										
											2018-09-22 22:51:54 -04:00
										 |  |  |         return ( | 
					
						
							| 
									
										
										
										
											2021-07-21 09:45:15 +02:00
										 |  |  |                 self.has_group("Bottles", player) and | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |                 self.has('Bug Catching Net', player) and | 
					
						
							| 
									
										
										
										
											2021-02-27 17:11:54 +01:00
										 |  |  |                 (self.has('Pegasus Boots', player) or (self.has_sword(player) and self.has('Quake', player))) and | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |                 cave.can_reach(self) and | 
					
						
							|  |  |  |                 self.is_not_bunny(cave, player) | 
					
						
							| 
									
										
										
										
											2018-09-22 22:51:54 -04:00
										 |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |     def can_retrieve_tablet(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2020-10-07 19:51:46 +02:00
										 |  |  |         return self.has('Book of Mudora', player) and (self.has_beam_sword(player) or | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                                                        (self.multiworld.swordless[player] and | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                                                         self.has("Hammer", player))) | 
					
						
							| 
									
										
										
										
											2020-10-07 19:51:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def has_sword(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2020-08-20 20:13:00 +02:00
										 |  |  |         return self.has('Fighter Sword', player) \ | 
					
						
							|  |  |  |                or self.has('Master Sword', player) \ | 
					
						
							|  |  |  |                or self.has('Tempered Sword', player) \ | 
					
						
							|  |  |  |                or self.has('Golden Sword', player) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def has_beam_sword(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |         return self.has('Master Sword', player) or self.has('Tempered Sword', player) or self.has('Golden Sword', | 
					
						
							|  |  |  |                                                                                                   player) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-20 19:17:10 +02:00
										 |  |  |     def has_melee_weapon(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         return self.has_sword(player) or self.has('Hammer', player) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def has_fire_source(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         return self.has('Fire Rod', player) or self.has('Lamp', player) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def can_melt_things(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2020-10-07 19:51:46 +02:00
										 |  |  |         return self.has('Fire Rod', player) or \ | 
					
						
							|  |  |  |                (self.has('Bombos', player) and | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 (self.multiworld.swordless[player] or | 
					
						
							| 
									
										
										
										
											2020-10-07 19:51:46 +02:00
										 |  |  |                  self.has_sword(player))) | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def can_avoid_lasers(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2019-07-27 09:13:13 -04:00
										 |  |  |         return self.has('Mirror Shield', player) or self.has('Cane of Byrna', player) or self.has('Cape', player) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def is_not_bunny(self, region: Region, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2021-02-27 17:11:54 +01:00
										 |  |  |         if self.has('Moon Pearl', player): | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |             return True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         return region.is_light_world if self.multiworld.mode[player] != 'inverted' else region.is_dark_world | 
					
						
							| 
									
										
										
										
											2019-07-27 09:13:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def can_reach_light_world(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2019-09-21 21:59:16 -04:00
										 |  |  |         if True in [i.is_light_world for i in self.reachable_regions[player]]: | 
					
						
							|  |  |  |             return True | 
					
						
							|  |  |  |         return False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def can_reach_dark_world(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2019-09-21 21:59:16 -04:00
										 |  |  |         if True in [i.is_dark_world for i in self.reachable_regions[player]]: | 
					
						
							|  |  |  |             return True | 
					
						
							|  |  |  |         return False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def has_misery_mire_medallion(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         return self.has(self.multiworld.required_medallions[player][0], player) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def has_turtle_rock_medallion(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         return self.has(self.multiworld.required_medallions[player][1], player) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def can_boots_clip_lw(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         if self.multiworld.mode[player] == 'inverted': | 
					
						
							| 
									
										
										
										
											2021-02-27 17:11:54 +01:00
										 |  |  |             return self.has('Pegasus Boots', player) and self.has('Moon Pearl', player) | 
					
						
							|  |  |  |         return self.has('Pegasus Boots', player) | 
					
						
							| 
									
										
										
										
											2020-02-10 00:38:55 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def can_boots_clip_dw(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         if self.multiworld.mode[player] != 'inverted': | 
					
						
							| 
									
										
										
										
											2021-02-27 17:11:54 +01:00
										 |  |  |             return self.has('Pegasus Boots', player) and self.has('Moon Pearl', player) | 
					
						
							|  |  |  |         return self.has('Pegasus Boots', player) | 
					
						
							| 
									
										
										
										
											2020-02-10 00:38:55 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def can_get_glitched_speed_lw(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2021-02-27 17:11:54 +01:00
										 |  |  |         rules = [self.has('Pegasus Boots', player), any([self.has('Hookshot', player), self.has_sword(player)])] | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         if self.multiworld.mode[player] == 'inverted': | 
					
						
							| 
									
										
										
										
											2021-02-27 17:11:54 +01:00
										 |  |  |             rules.append(self.has('Moon Pearl', player)) | 
					
						
							| 
									
										
										
										
											2020-02-10 23:54:35 -04:00
										 |  |  |         return all(rules) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def can_superbunny_mirror_with_sword(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2021-02-27 17:11:54 +01:00
										 |  |  |         return self.has('Magic Mirror', player) and self.has_sword(player) | 
					
						
							| 
									
										
										
										
											2020-02-12 19:48:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def can_get_glitched_speed_dw(self, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2021-02-27 17:11:54 +01:00
										 |  |  |         rules = [self.has('Pegasus Boots', player), any([self.has('Hookshot', player), self.has_sword(player)])] | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         if self.multiworld.mode[player] != 'inverted': | 
					
						
							| 
									
										
										
										
											2021-02-27 17:11:54 +01:00
										 |  |  |             rules.append(self.has('Moon Pearl', player)) | 
					
						
							| 
									
										
										
										
											2020-02-10 23:54:35 -04:00
										 |  |  |         return all(rules) | 
					
						
							| 
									
										
										
										
											2020-02-10 00:38:55 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |     def can_bomb_clip(self, region: Region, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2021-06-07 01:19:27 -05:00
										 |  |  |         return self.is_not_bunny(region, player) and self.has('Pegasus Boots', player) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-27 16:47:47 -07:00
										 |  |  |     def collect(self, item: Item, event: bool = False, location: Optional[Location] = None) -> bool: | 
					
						
							| 
									
										
										
										
											2018-01-01 15:55:13 -05:00
										 |  |  |         if location: | 
					
						
							|  |  |  |             self.locations_checked.add(location) | 
					
						
							| 
									
										
										
										
											2021-02-26 21:03:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         changed = self.multiworld.worlds[item.player].collect(self, item) | 
					
						
							| 
									
										
										
										
											2021-07-04 15:47:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if not changed and event: | 
					
						
							| 
									
										
										
										
											2020-03-07 23:35:55 +01:00
										 |  |  |             self.prog_items[item.name, item.player] += 1 | 
					
						
							| 
									
										
										
										
											2017-05-26 09:55:49 +02:00
										 |  |  |             changed = True | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-11 00:18:30 -04:00
										 |  |  |         self.stale[item.player] = True | 
					
						
							| 
									
										
										
										
											2017-05-26 09:55:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-14 17:52:01 +01:00
										 |  |  |         if changed and not event: | 
					
						
							|  |  |  |             self.sweep_for_events() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return changed | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-30 05:33:56 +01:00
										 |  |  |     def remove(self, item: Item): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         changed = self.multiworld.worlds[item.player].remove(self, item) | 
					
						
							| 
									
										
										
										
											2021-08-10 09:47:28 +02:00
										 |  |  |         if changed: | 
					
						
							|  |  |  |             # invalidate caches, nothing can be trusted anymore now | 
					
						
							|  |  |  |             self.reachable_regions[item.player] = set() | 
					
						
							|  |  |  |             self.blocked_connections[item.player] = set() | 
					
						
							|  |  |  |             self.stale[item.player] = True | 
					
						
							| 
									
										
										
										
											2017-05-16 21:23:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 21:51:43 -05:00
										 |  |  | @unique | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  | class RegionType(IntEnum): | 
					
						
							| 
									
										
										
										
											2021-07-15 08:50:08 +02:00
										 |  |  |     Generic = 0 | 
					
						
							| 
									
										
										
										
											2018-01-18 21:51:43 -05:00
										 |  |  |     LightWorld = 1 | 
					
						
							|  |  |  |     DarkWorld = 2 | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |     Cave = 3  # Also includes Houses | 
					
						
							| 
									
										
										
										
											2018-01-18 21:51:43 -05:00
										 |  |  |     Dungeon = 4 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							| 
									
										
										
										
											2021-11-30 05:33:56 +01:00
										 |  |  |     def is_indoors(self) -> bool: | 
					
						
							| 
									
										
										
										
											2018-01-18 21:51:43 -05:00
										 |  |  |         """Shorthand for checking if Cave or Dungeon""" | 
					
						
							|  |  |  |         return self in (RegionType.Cave, RegionType.Dungeon) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-20 19:19:56 +01:00
										 |  |  | class Region: | 
					
						
							|  |  |  |     name: str | 
					
						
							|  |  |  |     type: RegionType | 
					
						
							|  |  |  |     hint_text: str | 
					
						
							|  |  |  |     player: int | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |     multiworld: Optional[MultiWorld] | 
					
						
							| 
									
										
										
										
											2022-02-20 19:19:56 +01:00
										 |  |  |     entrances: List[Entrance] | 
					
						
							|  |  |  |     exits: List[Entrance] | 
					
						
							|  |  |  |     locations: List[Location] | 
					
						
							|  |  |  |     dungeon: Optional[Dungeon] = None | 
					
						
							|  |  |  |     shop: Optional = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # LttP specific. TODO: move to a LttPRegion | 
					
						
							|  |  |  |     # will be set after making connections. | 
					
						
							|  |  |  |     is_light_world: bool = False | 
					
						
							|  |  |  |     is_dark_world: bool = False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def __init__(self, name: str, type_: RegionType, hint: str, player: int, world: Optional[MultiWorld] = None): | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         self.name = name | 
					
						
							| 
									
										
										
										
											2022-01-06 06:09:15 +01:00
										 |  |  |         self.type = type_ | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         self.entrances = [] | 
					
						
							|  |  |  |         self.exits = [] | 
					
						
							| 
									
										
										
										
											2022-02-20 19:19:56 +01:00
										 |  |  |         self.locations = [] | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         self.multiworld = world | 
					
						
							| 
									
										
										
										
											2019-01-20 01:01:02 -06:00
										 |  |  |         self.hint_text = hint | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         self.player = player | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-30 05:33:56 +01:00
										 |  |  |     def can_reach(self, state: CollectionState) -> bool: | 
					
						
							| 
									
										
										
										
											2019-07-11 00:18:30 -04:00
										 |  |  |         if state.stale[self.player]: | 
					
						
							|  |  |  |             state.update_reachable_regions(self.player) | 
					
						
							|  |  |  |         return self in state.reachable_regions[self.player] | 
					
						
							| 
									
										
										
										
											2019-07-08 22:48:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-30 05:33:56 +01:00
										 |  |  |     def can_reach_private(self, state: CollectionState) -> bool: | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         for entrance in self.entrances: | 
					
						
							| 
									
										
										
										
											2019-07-11 00:18:30 -04:00
										 |  |  |             if entrance.can_reach(state): | 
					
						
							| 
									
										
										
										
											2018-01-01 15:55:13 -05:00
										 |  |  |                 if not self in state.path: | 
					
						
							|  |  |  |                     state.path[self] = (self.name, state.path.get(entrance, None)) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |                 return True | 
					
						
							|  |  |  |         return False | 
					
						
							| 
									
										
										
										
											2017-10-28 18:34:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-18 14:30:43 +02:00
										 |  |  |     def get_connecting_entrance(self, is_main_entrance: typing.Callable[[Entrance], bool]) -> Entrance: | 
					
						
							|  |  |  |         for entrance in self.entrances: | 
					
						
							|  |  |  |             if is_main_entrance(entrance): | 
					
						
							|  |  |  |                 return entrance | 
					
						
							|  |  |  |         for entrance in self.entrances:  # BFS might be better here, trying DFS for now. | 
					
						
							|  |  |  |             return entrance.parent_region.get_connecting_entrance(is_main_entrance) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-10 21:31:15 +02:00
										 |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return self.__str__() | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-10 21:31:15 +02:00
										 |  |  |     def __str__(self): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         return self.multiworld.get_name_string_for_object(self) if self.multiworld else f'{self.name} (Player {self.player})' | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-20 19:10:08 +01:00
										 |  |  | class Entrance: | 
					
						
							|  |  |  |     access_rule: Callable[[CollectionState], bool] = staticmethod(lambda state: True) | 
					
						
							|  |  |  |     hide_path: bool = False | 
					
						
							|  |  |  |     player: int | 
					
						
							|  |  |  |     name: str | 
					
						
							|  |  |  |     parent_region: Optional[Region] | 
					
						
							|  |  |  |     connected_region: Optional[Region] = None | 
					
						
							|  |  |  |     # LttP specific, TODO: should make a LttPEntrance | 
					
						
							|  |  |  |     addresses = None | 
					
						
							|  |  |  |     target = None | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def __init__(self, player: int, name: str = '', parent: Region = None): | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         self.name = name | 
					
						
							|  |  |  |         self.parent_region = parent | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         self.player = player | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-30 05:33:56 +01:00
										 |  |  |     def can_reach(self, state: CollectionState) -> bool: | 
					
						
							| 
									
										
										
										
											2019-07-11 00:18:30 -04:00
										 |  |  |         if self.parent_region.can_reach(state) and self.access_rule(state): | 
					
						
							| 
									
										
										
										
											2020-05-10 19:27:13 +10:00
										 |  |  |             if not self.hide_path and not self in state.path: | 
					
						
							| 
									
										
										
										
											2018-01-01 15:55:13 -05:00
										 |  |  |                 state.path[self] = (self.name, state.path.get(self.parent_region, (self.parent_region.name, None))) | 
					
						
							| 
									
										
										
										
											2017-05-20 14:03:15 +02:00
										 |  |  |             return True | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-20 14:03:15 +02:00
										 |  |  |         return False | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-28 14:54:10 -07:00
										 |  |  |     def connect(self, region: Region, addresses: Any = None, target: Any = None) -> None: | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         self.connected_region = region | 
					
						
							| 
									
										
										
										
											2017-05-20 14:03:15 +02:00
										 |  |  |         self.target = target | 
					
						
							| 
									
										
										
										
											2017-06-03 15:33:11 +02:00
										 |  |  |         self.addresses = addresses | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         region.entrances.append(self) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-10 21:31:15 +02:00
										 |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return self.__str__() | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-10 21:31:15 +02:00
										 |  |  |     def __str__(self): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         world = self.parent_region.multiworld if self.parent_region else None | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         return world.get_name_string_for_object(self) if world else f'{self.name} (Player {self.player})' | 
					
						
							| 
									
										
										
										
											2017-10-28 18:34:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-15 12:16:07 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-30 16:31:56 +02:00
										 |  |  | class Dungeon(object): | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def __init__(self, name: str, regions: List[Region], big_key: Item, small_keys: List[Item], | 
					
						
							|  |  |  |                  dungeon_items: List[Item], player: int): | 
					
						
							| 
									
										
										
										
											2017-10-15 12:16:07 -04:00
										 |  |  |         self.name = name | 
					
						
							|  |  |  |         self.regions = regions | 
					
						
							|  |  |  |         self.big_key = big_key | 
					
						
							|  |  |  |         self.small_keys = small_keys | 
					
						
							|  |  |  |         self.dungeon_items = dungeon_items | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |         self.bosses = dict() | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         self.player = player | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         self.multiworld = None | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							| 
									
										
										
										
											2021-11-30 05:33:56 +01:00
										 |  |  |     def boss(self) -> Optional[Boss]: | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |         return self.bosses.get(None, None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @boss.setter | 
					
						
							| 
									
										
										
										
											2021-11-30 05:33:56 +01:00
										 |  |  |     def boss(self, value: Optional[Boss]): | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |         self.bosses[None] = value | 
					
						
							| 
									
										
										
										
											2017-10-28 18:34:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-15 12:16:07 -04:00
										 |  |  |     @property | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def keys(self) -> List[Item]: | 
					
						
							| 
									
										
										
										
											2017-10-15 12:16:07 -04:00
										 |  |  |         return self.small_keys + ([self.big_key] if self.big_key else []) | 
					
						
							| 
									
										
										
										
											2017-10-28 18:34:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-15 12:16:07 -04:00
										 |  |  |     @property | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def all_items(self) -> List[Item]: | 
					
						
							| 
									
										
										
										
											2017-10-28 18:34:37 -04:00
										 |  |  |         return self.dungeon_items + self.keys | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-04 03:30:59 +02:00
										 |  |  |     def is_dungeon_item(self, item: Item) -> bool: | 
					
						
							| 
									
										
										
										
											2021-03-21 00:47:17 +01:00
										 |  |  |         return item.player == self.player and item.name in (dungeon_item.name for dungeon_item in self.all_items) | 
					
						
							| 
									
										
										
										
											2017-10-28 18:34:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |     def __eq__(self, other: Dungeon) -> bool: | 
					
						
							|  |  |  |         if not other: | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2020-06-04 03:30:59 +02:00
										 |  |  |         return self.name == other.name and self.player == other.player | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-10 21:31:15 +02:00
										 |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return self.__str__() | 
					
						
							| 
									
										
										
										
											2017-10-15 12:16:07 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-10 21:31:15 +02:00
										 |  |  |     def __str__(self): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         return self.multiworld.get_name_string_for_object(self) if self.multiworld else f'{self.name} (Player {self.player})' | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-10 19:23:57 +01:00
										 |  |  | class Boss(): | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def __init__(self, name: str, enemizer_name: str, defeat_rule: Callable, player: int): | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |         self.name = name | 
					
						
							|  |  |  |         self.enemizer_name = enemizer_name | 
					
						
							|  |  |  |         self.defeat_rule = defeat_rule | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         self.player = player | 
					
						
							| 
									
										
										
										
											2017-10-28 18:34:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-03 00:12:14 +01:00
										 |  |  |     def can_defeat(self, state) -> bool: | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         return self.defeat_rule(state, self.player) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 04:05:36 +01:00
										 |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return f"Boss({self.name})" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  | class LocationProgressType(IntEnum): | 
					
						
							| 
									
										
										
										
											2022-01-19 20:19:07 -07:00
										 |  |  |     DEFAULT = 1 | 
					
						
							|  |  |  |     PRIORITY = 2 | 
					
						
							|  |  |  |     EXCLUDED = 3 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-01 16:36:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-20 19:19:56 +01:00
										 |  |  | class Location: | 
					
						
							| 
									
										
										
										
											2022-08-04 14:10:58 +02:00
										 |  |  |     game: str = "Generic" | 
					
						
							|  |  |  |     player: int | 
					
						
							|  |  |  |     name: str | 
					
						
							|  |  |  |     address: Optional[int] | 
					
						
							|  |  |  |     parent_region: Optional[Region] | 
					
						
							| 
									
										
										
										
											2021-01-10 19:23:57 +01:00
										 |  |  |     event: bool = False | 
					
						
							|  |  |  |     locked: bool = False | 
					
						
							| 
									
										
											  
											
												Ocarina of Time (#64)
* first commit (not including OoT data files yet)
* added some basic options
* rule parser works now at least
* make sure to commit everything this time
* temporary change to BaseClasses for oot
* overworld location graph builds mostly correctly
* adding oot data files
* commenting out world options until later since they only existed to make the RuleParser work
* conversion functions between AP ids and OOT ids
* world graph outputs
* set scrub prices
* itempool generates, entrances connected, way too many options added
* fixed set_rules and set_shop_rules
* temp baseclasses changes
* Reaches the fill step now, old event-based system retained in case the new way breaks
* Song placements and misc fixes everywhere
* temporary changes to make oot work
* changed root exits for AP fill framework
* prevent infinite recursion due to OoT sharing usage of the address field
* age reachability works hopefully, songs are broken again
* working spoiler log generation on beatable-only
* Logic tricks implemented
* need this for logic tricks
* fixed map/compass being placed on Serenade location
* kill unreachable events before filling the world
* add a bunch of utility functions to prepare for rom patching
* move OptionList into generic options
* fixed some silly bugs with OptionList
* properly seed all random behavior (so far)
* ROM generation working
* fix hints trying to get alttp dungeon hint texts
* continue fixing hints
* add oot to network data package
* change item and location IDs to 66000 and 67000 range respectively
* push removed items to precollected items
* fixed various issues with cross-contamination with multiple world generation
* reenable glitched logic (hopefully)
* glitched world files age-check fix
* cleaned up some get_locations calls
* added token shuffle and scrub shuffle, modified some options slightly to make the parsing work
* reenable MQ dungeons
* fix forest mq exception
* made targeting style an option for now, will be cosmetic later
* reminder to move targeting to cosmetics
* some oot option maintenance
* enabled starting time of day
* fixed issue breaking shop slots in multiworld generation
* added "off" option for text shuffle and hints
* shopsanity functionality restored
* change patch file extension
* remove unnecessary utility functions + imports
* update MIT license
* change option to "patch_uncompressed_rom" instead of "compress_rom"
* compliance with new AutoWorld systems
* Kill only internal events, remove non-internal big poe event in code
* re-add the big poe event and handle it correctly
* remove extra method in Range option
* fix typo
* Starting items, starting with consumables option
* do not remove nonexistent item
* move set_shop_rules to after shop items are placed
* some cleanup
* add retries for song placement
* flagged Skull Mask and Mask of Truth as advancement items
* update OoT to use LogicMixin
* Fixed trying to assign starting items from the wrong players
* fixed song retry step
* improved option handling, comments, and starting item replacements
* DefaultOnToggle writes Yes or No to spoiler
* enable compression of output if Compress executable is present
* clean up compression
* check whether (de)compressor exists before running the process
* allow specification of rom path in host.yaml
* check if decompressed file already exists before decompressing again
* fix triforce hunt generation
* rename all the oot state functions with prefix
* OoT: mark triforce pieces as completion goal for triforce hunt
* added overworld and any-dungeon shuffle for dungeon items
* Hide most unshuffled locations and events from the list of locations in spoiler
* build oot option ranges with a generic function instead of defining each separately
* move oot output-type control to host.yaml instead of individual yamls
* implement dungeon song shuffle
* minor improvements to overworld dungeon item shuffle
* remove random ice trap names in shops, mostly to avoid maintaining a massive censor list
* always output patch file to folder, remove option to generate ROM in preparation for removal
* re-add the fix for infinite recursion due to not being light or dark world
* change AP-sendable to Ocarina of Time model, since the triforce piece has some extra code apparently
* oot: remove item_names and location_names
* oot: minor fixes
* oot: comment out ROM patching
* oot: only add CollectionState objects on creation if actually needed
* main entrance shuffle method and entrances-based rules
* fix entrances based rules
* disable master quest and big poe count options for client compatibility
* use get_player_name instead of get_player_names
* fix OptionList
* fix oot options for new option system
* new coop section in oot rom: expand player names to 16 bytes, write AP_PLAYER_NAME at end of PLAYER_NAMES
* fill AP player name in oot rom with 0 instead of 0xDF
* encode player name with ASCII for fixed-width
* revert oot player name array to 8 bytes per name
* remove Pierre location if fast scarecrow is on
* check player name length
* "free_scarecrow" not "fast_scarecrow"
* OoT locations now properly store the AP ID instead of the oot internal ID
* oot __version__ updates in lockstep with AP version
* pull in unmodified oot cosmetic files
* also grab JSONDump since it's needed apparently
* gather extra needed methods, modify imports
* delete cosmetics log, replace all instances of SettingsList with OOTWorld
* cosmetic options working, except for sound effects (due to ear-safe issues)
* SFX, Music, and Fanfare randomization reenabled
* move OoT data files into the worlds folder
* move Compress and Decompress into oot data folder
* Replace get_all_state with custom method to avoid the cache
* OoT ROM: increment item counter before setting incoming item/player values to 0, preventing desync issues
* set data_version to 0
* make Kokiri Sword shuffle off by default
* reenable "Random Choice" for various cosmetic options
* kill Ruto's Letter turnin if open fountain
also fix for shopsanity
* place Buy Goron/Zora Tunic first in shop shuffle
* make ice traps appear as other items instead of breaking generation
* managed to break ice traps on non-major-only
* only handle ice traps if they are on
* fix shopsanity for non-oot games, and write player name instead of player number
* light arrows hint uses player name instead of player number
* Reenable "skip child zelda" option
* fix entrances_based_rules
* fix ganondorf hint if starting with light arrows
* fix dungeonitem shuffle and shopsanity interaction
* remove has_all_of, has_any_of, count_of in BaseClasses, replace usage with has_all, has_any, has_group
* force local giveable item on ZL if skip_child_zelda and shuffle_song_items is any
* keep bosses and bombchu bowling chus out of data package
* revert workaround for infinite recursion and fix it properly
* fix shared shop id caches during patching process
* fix shop text box overflows, as much as possible
* add default oot host.yaml option
* add .apz5, .n64, .z64 to gitignore
* Properly document and name all (functioning) OOT options
* clean up some imports
* remove unnecessary files from oot's data
* fix typo in gitignore
* readd the Compress and Decompress utilities, since they are needed for generation
* cleanup of imports and some minor optimizations
* increase shop offset for item IDs to 0xCB
* remove shop item AP ids entirely
* prevent triforce pieces for other players from being received by yourself
* add "excluded" property to Location
* Hint system adapted and reenabled; hints still unseeded
* make hints deterministic with lists instead of sets
* do not allow hints to point to Light Arrows on non-vanilla bridge
* foreign locations hint as their full name in OoT rather than their region
* checkedLocations now stores hint names by player ID, so that the same location in different worlds can have hints associated
* consolidate versioning in Utils
* ice traps appear as major items rather than any progression item
* set prescription and claim check as defaults for adult trade item settings
* add oot options to playerSettings
* allow case-insensitive logic tricks in yaml
* fix oot shopsanity option formatting
* Write OoT override info even if local item, enabling local checks to show up immediately in the client
* implement CollectionState.can_live_dmg for oot glitched logic
* filter item names for invalid characters when patching shops
* make ice traps appear according to the settings of the world they are shuffled into, rather than the original world
* set hidden-spoiler items and locations with Shop items to events
* make GF carpenters, Gerudo Card, Malon, ZL, and Impa events if the relevant settings are enabled, preventing them from appearing in the client on game start
* Fix oot Glitched and No Logic generation
* fix indenting
* Greatly reduce displayed cosmetic options
* Change oot data version to 1
* add apz5 distribution to webhost
* print player name if an ALttP dungeon contains a good item for OoT world
* delete unneeded commented code
* remove OcarinaSongs import to satisfy lint
											
										 
											2021-09-02 08:35:05 -04:00
										 |  |  |     show_in_spoiler: bool = True | 
					
						
							| 
									
										
										
										
											2022-01-19 20:19:07 -07:00
										 |  |  |     progress_type: LocationProgressType = LocationProgressType.DEFAULT | 
					
						
							| 
									
										
										
										
											2021-07-15 08:50:08 +02:00
										 |  |  |     always_allow = staticmethod(lambda item, state: False) | 
					
						
							| 
									
										
										
										
											2022-09-28 14:54:10 -07:00
										 |  |  |     access_rule: Callable[[CollectionState], bool] = staticmethod(lambda state: True) | 
					
						
							| 
									
										
										
										
											2021-07-15 08:50:08 +02:00
										 |  |  |     item_rule = staticmethod(lambda item: True) | 
					
						
							| 
									
										
										
										
											2022-02-20 19:19:56 +01:00
										 |  |  |     item: Optional[Item] = None | 
					
						
							| 
									
										
										
										
											2021-01-10 19:23:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-04 14:10:58 +02:00
										 |  |  |     def __init__(self, player: int, name: str = '', address: Optional[int] = None, parent: Optional[Region] = None): | 
					
						
							|  |  |  |         self.player = player | 
					
						
							|  |  |  |         self.name = name | 
					
						
							|  |  |  |         self.address = address | 
					
						
							| 
									
										
										
										
											2022-02-20 21:54:00 +01:00
										 |  |  |         self.parent_region = parent | 
					
						
							| 
									
										
										
										
											2017-10-28 18:34:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-09 16:16:31 +02:00
										 |  |  |     def can_fill(self, state: CollectionState, item: Item, check_access=True) -> bool: | 
					
						
							| 
									
										
										
										
											2022-10-31 00:47:23 +01:00
										 |  |  |         return (self.always_allow(state, item) | 
					
						
							|  |  |  |                 or ((self.progress_type != LocationProgressType.EXCLUDED or not (item.advancement or item.useful)) | 
					
						
							|  |  |  |                     and self.item_rule(item) | 
					
						
							|  |  |  |                     and (not check_access or self.can_reach(state)))) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-09 16:16:31 +02:00
										 |  |  |     def can_reach(self, state: CollectionState) -> bool: | 
					
						
							| 
									
										
										
										
											2020-08-21 18:35:48 +02:00
										 |  |  |         # self.access_rule computes faster on average, so placing it first for faster abort | 
					
						
							| 
									
										
										
										
											2022-10-31 00:47:23 +01:00
										 |  |  |         assert self.parent_region, "Can't reach location without region" | 
					
						
							|  |  |  |         return self.access_rule(state) and self.parent_region.can_reach(state) | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-06 16:08:17 +02:00
										 |  |  |     def place_locked_item(self, item: Item): | 
					
						
							|  |  |  |         if self.item: | 
					
						
							|  |  |  |             raise Exception(f"Location {self} already filled.") | 
					
						
							|  |  |  |         self.item = item | 
					
						
							| 
									
										
										
										
											2022-03-25 22:57:00 +01:00
										 |  |  |         item.location = self | 
					
						
							| 
									
										
										
										
											2021-06-06 16:08:17 +02:00
										 |  |  |         self.event = item.advancement | 
					
						
							|  |  |  |         self.locked = True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-10 21:31:15 +02:00
										 |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return self.__str__() | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-10 21:31:15 +02:00
										 |  |  |     def __str__(self): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         world = self.parent_region.multiworld if self.parent_region and self.parent_region.multiworld else None | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         return world.get_name_string_for_object(self) if world else f'{self.name} (Player {self.player})' | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-22 19:19:29 +02:00
										 |  |  |     def __hash__(self): | 
					
						
							|  |  |  |         return hash((self.name, self.player)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def __lt__(self, other: Location): | 
					
						
							| 
									
										
										
										
											2021-01-17 22:58:52 +01:00
										 |  |  |         return (self.player, self.name) < (other.player, other.name) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 14:10:49 +02:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def native_item(self) -> bool: | 
					
						
							|  |  |  |         """Returns True if the item in this location matches game.""" | 
					
						
							|  |  |  |         return self.item and self.item.game == self.game | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |     @property | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def hint_text(self) -> str: | 
					
						
							| 
									
										
										
										
											2021-08-28 23:18:45 +02:00
										 |  |  |         hint_text = getattr(self, "_hint_text", None) | 
					
						
							|  |  |  |         if hint_text: | 
					
						
							|  |  |  |             return hint_text | 
					
						
							|  |  |  |         return "at " + self.name.replace("_", " ").replace("-", " ") | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-20 18:23:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  | class ItemClassification(IntFlag): | 
					
						
							|  |  |  |     filler = 0b0000  # aka trash, as in filler items like ammo, currency etc, | 
					
						
							|  |  |  |     progression = 0b0001  # Item that is logically relevant | 
					
						
							|  |  |  |     useful = 0b0010  # Item that is generally quite useful, but not required for anything logical | 
					
						
							|  |  |  |     trap = 0b0100  # detrimental or entirely useless (nothing) item | 
					
						
							|  |  |  |     skip_balancing = 0b1000  # should technically never occur on its own | 
					
						
							|  |  |  |     # Item that is logically relevant, but progression balancing should not touch. | 
					
						
							|  |  |  |     # Typically currency or other counted items. | 
					
						
							|  |  |  |     progression_skip_balancing = 0b1001  # only progression gets balanced | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def as_flag(self) -> int: | 
					
						
							|  |  |  |         """As Network API flag int.""" | 
					
						
							| 
									
										
										
										
											2022-06-17 06:10:30 +02:00
										 |  |  |         return int(self & 0b0111) | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Item: | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |     game: str = "Generic" | 
					
						
							| 
									
										
										
										
											2022-08-06 00:49:54 +02:00
										 |  |  |     __slots__ = ("name", "classification", "code", "player", "location") | 
					
						
							|  |  |  |     name: str | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  |     classification: ItemClassification | 
					
						
							| 
									
										
										
										
											2022-08-06 00:49:54 +02:00
										 |  |  |     code: Optional[int] | 
					
						
							|  |  |  |     """an item with code None is called an Event, and does not get written to multidata""" | 
					
						
							|  |  |  |     player: int | 
					
						
							|  |  |  |     location: Optional[Location] | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  |     def __init__(self, name: str, classification: ItemClassification, code: Optional[int], player: int): | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  |         self.name = name | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  |         self.classification = classification | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         self.player = player | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |         self.code = code | 
					
						
							| 
									
										
										
										
											2022-08-06 00:49:54 +02:00
										 |  |  |         self.location = None | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							| 
									
										
										
										
											2022-08-05 17:09:21 +02:00
										 |  |  |     def hint_text(self) -> str: | 
					
						
							| 
									
										
										
										
											2021-06-14 02:20:13 +02:00
										 |  |  |         return getattr(self, "_hint_text", self.name.replace("_", " ").replace("-", " ")) | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							| 
									
										
										
										
											2022-08-05 17:09:21 +02:00
										 |  |  |     def pedestal_hint_text(self) -> str: | 
					
						
							| 
									
										
										
										
											2021-06-14 02:23:41 +02:00
										 |  |  |         return getattr(self, "_pedestal_hint_text", self.name.replace("_", " ").replace("-", " ")) | 
					
						
							| 
									
										
										
										
											2017-10-28 18:34:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def advancement(self) -> bool: | 
					
						
							| 
									
										
										
										
											2022-07-02 13:27:50 +02:00
										 |  |  |         return ItemClassification.progression in self.classification | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def skip_in_prog_balancing(self) -> bool: | 
					
						
							| 
									
										
										
										
											2022-07-02 13:27:50 +02:00
										 |  |  |         return ItemClassification.progression_skip_balancing in self.classification | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def useful(self) -> bool: | 
					
						
							| 
									
										
										
										
											2022-07-02 13:27:50 +02:00
										 |  |  |         return ItemClassification.useful in self.classification | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def trap(self) -> bool: | 
					
						
							| 
									
										
										
										
											2022-07-02 13:27:50 +02:00
										 |  |  |         return ItemClassification.trap in self.classification | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-18 06:16:16 +01:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def flags(self) -> int: | 
					
						
							| 
									
										
										
										
											2022-06-17 06:10:30 +02:00
										 |  |  |         return self.classification.as_flag() | 
					
						
							| 
									
										
										
										
											2022-01-18 06:16:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-02 12:49:43 +01:00
										 |  |  |     def __eq__(self, other): | 
					
						
							|  |  |  |         return self.name == other.name and self.player == other.player | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-05 17:09:21 +02:00
										 |  |  |     def __lt__(self, other: Item) -> bool: | 
					
						
							| 
									
										
										
										
											2021-02-03 14:24:29 +01:00
										 |  |  |         if other.player != self.player: | 
					
						
							|  |  |  |             return other.player < self.player | 
					
						
							|  |  |  |         return self.name < other.name | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-02 12:59:19 +01:00
										 |  |  |     def __hash__(self): | 
					
						
							|  |  |  |         return hash((self.name, self.player)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-05 17:09:21 +02:00
										 |  |  |     def __repr__(self) -> str: | 
					
						
							| 
									
										
										
										
											2020-04-10 21:31:15 +02:00
										 |  |  |         return self.__str__() | 
					
						
							| 
									
										
										
										
											2017-05-15 20:28:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-05 17:09:21 +02:00
										 |  |  |     def __str__(self) -> str: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         if self.location and self.location.parent_region and self.location.parent_region.multiworld: | 
					
						
							|  |  |  |             return self.location.parent_region.multiworld.get_name_string_for_object(self) | 
					
						
							| 
									
										
										
										
											2022-08-05 17:09:21 +02:00
										 |  |  |         return f"{self.name} (Player {self.player})" | 
					
						
							| 
									
										
										
										
											2017-05-20 14:03:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-02 04:57:57 +02:00
										 |  |  | class Spoiler(): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |     multiworld: MultiWorld | 
					
						
							| 
									
										
										
										
											2022-02-07 00:26:44 +01:00
										 |  |  |     unreachables: Set[Location] | 
					
						
							| 
									
										
										
										
											2020-08-25 14:31:20 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  |     def __init__(self, world): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         self.multiworld = world | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         self.hashes = {} | 
					
						
							| 
									
										
										
										
											2018-03-24 01:50:54 -04:00
										 |  |  |         self.entrances = OrderedDict() | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  |         self.medallions = {} | 
					
						
							|  |  |  |         self.playthrough = {} | 
					
						
							| 
									
										
										
										
											2022-02-07 00:26:44 +01:00
										 |  |  |         self.unreachables = set() | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  |         self.locations = {} | 
					
						
							| 
									
										
										
										
											2018-01-01 15:55:13 -05:00
										 |  |  |         self.paths = {} | 
					
						
							| 
									
										
										
										
											2018-03-24 01:43:10 -04:00
										 |  |  |         self.shops = [] | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |         self.bosses = OrderedDict() | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def set_entrance(self, entrance: str, exit_: str, direction: str, player: int): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         if self.multiworld.players == 1: | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |             self.entrances[(entrance, direction, player)] = OrderedDict( | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |                 [('entrance', entrance), ('exit', exit_), ('direction', direction)]) | 
					
						
							| 
									
										
										
										
											2019-07-13 18:11:43 -04:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |             self.entrances[(entrance, direction, player)] = OrderedDict( | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |                 [('player', player), ('entrance', entrance), ('exit', exit_), ('direction', direction)]) | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def parse_data(self): | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         self.medallions = OrderedDict() | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         for player in self.multiworld.get_game_players("A Link to the Past"): | 
					
						
							|  |  |  |             self.medallions[f'Misery Mire ({self.multiworld.get_player_name(player)})'] = \ | 
					
						
							|  |  |  |                 self.multiworld.required_medallions[player][0] | 
					
						
							|  |  |  |             self.medallions[f'Turtle Rock ({self.multiworld.get_player_name(player)})'] = \ | 
					
						
							|  |  |  |                 self.multiworld.required_medallions[player][1] | 
					
						
							| 
									
										
										
										
											2018-03-23 11:03:38 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.locations = OrderedDict() | 
					
						
							|  |  |  |         listed_locations = set() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         lw_locations = [loc for loc in self.multiworld.get_locations() if | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                         loc not in listed_locations and loc.parent_region and loc.parent_region.type == RegionType.LightWorld and loc.show_in_spoiler] | 
					
						
							|  |  |  |         self.locations['Light World'] = OrderedDict( | 
					
						
							|  |  |  |             [(str(location), str(location.item) if location.item is not None else 'Nothing') for location in | 
					
						
							|  |  |  |              lw_locations]) | 
					
						
							| 
									
										
										
										
											2018-03-23 11:03:38 -04:00
										 |  |  |         listed_locations.update(lw_locations) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         dw_locations = [loc for loc in self.multiworld.get_locations() if | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                         loc not in listed_locations and loc.parent_region and loc.parent_region.type == RegionType.DarkWorld and loc.show_in_spoiler] | 
					
						
							|  |  |  |         self.locations['Dark World'] = OrderedDict( | 
					
						
							|  |  |  |             [(str(location), str(location.item) if location.item is not None else 'Nothing') for location in | 
					
						
							|  |  |  |              dw_locations]) | 
					
						
							| 
									
										
										
										
											2018-03-23 11:03:38 -04:00
										 |  |  |         listed_locations.update(dw_locations) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         cave_locations = [loc for loc in self.multiworld.get_locations() if | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                           loc not in listed_locations and loc.parent_region and loc.parent_region.type == RegionType.Cave and loc.show_in_spoiler] | 
					
						
							|  |  |  |         self.locations['Caves'] = OrderedDict( | 
					
						
							|  |  |  |             [(str(location), str(location.item) if location.item is not None else 'Nothing') for location in | 
					
						
							|  |  |  |              cave_locations]) | 
					
						
							| 
									
										
										
										
											2018-03-23 11:03:38 -04:00
										 |  |  |         listed_locations.update(cave_locations) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         for dungeon in self.multiworld.dungeons.values(): | 
					
						
							|  |  |  |             dungeon_locations = [loc for loc in self.multiworld.get_locations() if | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                                  loc not in listed_locations and loc.parent_region and loc.parent_region.dungeon == dungeon and loc.show_in_spoiler] | 
					
						
							|  |  |  |             self.locations[str(dungeon)] = OrderedDict( | 
					
						
							|  |  |  |                 [(str(location), str(location.item) if location.item is not None else 'Nothing') for location in | 
					
						
							|  |  |  |                  dungeon_locations]) | 
					
						
							| 
									
										
										
										
											2018-03-23 11:03:38 -04:00
										 |  |  |             listed_locations.update(dungeon_locations) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         other_locations = [loc for loc in self.multiworld.get_locations() if | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                            loc not in listed_locations and loc.show_in_spoiler] | 
					
						
							| 
									
										
										
										
											2018-03-23 11:03:38 -04:00
										 |  |  |         if other_locations: | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |             self.locations['Other Locations'] = OrderedDict( | 
					
						
							|  |  |  |                 [(str(location), str(location.item) if location.item is not None else 'Nothing') for location in | 
					
						
							|  |  |  |                  other_locations]) | 
					
						
							| 
									
										
										
										
											2018-03-23 11:03:38 -04:00
										 |  |  |             listed_locations.update(other_locations) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         self.shops = [] | 
					
						
							| 
									
										
										
										
											2021-09-12 16:09:13 -05:00
										 |  |  |         from worlds.alttp.Shops import ShopType, price_type_display_name, price_rate_display | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         for shop in self.multiworld.shops: | 
					
						
							| 
									
										
										
										
											2020-01-10 11:41:22 +01:00
										 |  |  |             if not shop.custom: | 
					
						
							| 
									
										
										
										
											2018-03-24 01:43:10 -04:00
										 |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2021-09-12 20:25:08 +02:00
										 |  |  |             shopdata = { | 
					
						
							|  |  |  |                 'location': str(shop.region), | 
					
						
							|  |  |  |                 'type': 'Take Any' if shop.type == ShopType.TakeAny else 'Shop' | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-03-24 01:43:10 -04:00
										 |  |  |             for index, item in enumerate(shop.inventory): | 
					
						
							|  |  |  |                 if item is None: | 
					
						
							|  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2021-09-12 16:09:13 -05:00
										 |  |  |                 my_price = item['price'] // price_rate_display.get(item['price_type'], 1) | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                 shopdata['item_{}'.format( | 
					
						
							|  |  |  |                     index)] = f"{item['item']} — {my_price} {price_type_display_name[item['price_type']]}" | 
					
						
							| 
									
										
										
										
											2020-11-23 20:05:04 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if item['player'] > 0: | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                     shopdata['item_{}'.format(index)] = shopdata['item_{}'.format(index)].replace('—', | 
					
						
							|  |  |  |                                                                                                   '(Player {}) — '.format( | 
					
						
							|  |  |  |                                                                                                       item['player'])) | 
					
						
							| 
									
										
										
										
											2020-11-23 20:05:04 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-01 21:23:43 -07:00
										 |  |  |                 if item['max'] == 0: | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 shopdata['item_{}'.format(index)] += " x {}".format(item['max']) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if item['replacement'] is None: | 
					
						
							|  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                 shopdata['item_{}'.format( | 
					
						
							|  |  |  |                     index)] += f", {item['replacement']} - {item['replacement_price']} {price_type_display_name[item['replacement_price_type']]}" | 
					
						
							| 
									
										
										
										
											2018-03-24 01:43:10 -04:00
										 |  |  |             self.shops.append(shopdata) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         for player in self.multiworld.get_game_players("A Link to the Past"): | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |             self.bosses[str(player)] = OrderedDict() | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             self.bosses[str(player)]["Eastern Palace"] = self.multiworld.get_dungeon("Eastern Palace", player).boss.name | 
					
						
							|  |  |  |             self.bosses[str(player)]["Desert Palace"] = self.multiworld.get_dungeon("Desert Palace", player).boss.name | 
					
						
							|  |  |  |             self.bosses[str(player)]["Tower Of Hera"] = self.multiworld.get_dungeon("Tower of Hera", player).boss.name | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |             self.bosses[str(player)]["Hyrule Castle"] = "Agahnim" | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             self.bosses[str(player)]["Palace Of Darkness"] = self.multiworld.get_dungeon("Palace of Darkness", | 
					
						
							|  |  |  |                                                                                          player).boss.name | 
					
						
							|  |  |  |             self.bosses[str(player)]["Swamp Palace"] = self.multiworld.get_dungeon("Swamp Palace", player).boss.name | 
					
						
							|  |  |  |             self.bosses[str(player)]["Skull Woods"] = self.multiworld.get_dungeon("Skull Woods", player).boss.name | 
					
						
							|  |  |  |             self.bosses[str(player)]["Thieves Town"] = self.multiworld.get_dungeon("Thieves Town", player).boss.name | 
					
						
							|  |  |  |             self.bosses[str(player)]["Ice Palace"] = self.multiworld.get_dungeon("Ice Palace", player).boss.name | 
					
						
							|  |  |  |             self.bosses[str(player)]["Misery Mire"] = self.multiworld.get_dungeon("Misery Mire", player).boss.name | 
					
						
							|  |  |  |             self.bosses[str(player)]["Turtle Rock"] = self.multiworld.get_dungeon("Turtle Rock", player).boss.name | 
					
						
							|  |  |  |             if self.multiworld.mode[player] != 'inverted': | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                 self.bosses[str(player)]["Ganons Tower Basement"] = \ | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     self.multiworld.get_dungeon('Ganons Tower', player).bosses['bottom'].name | 
					
						
							|  |  |  |                 self.bosses[str(player)]["Ganons Tower Middle"] = self.multiworld.get_dungeon('Ganons Tower', player).bosses[ | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                     'middle'].name | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 self.bosses[str(player)]["Ganons Tower Top"] = self.multiworld.get_dungeon('Ganons Tower', player).bosses[ | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                     'top'].name | 
					
						
							| 
									
										
										
										
											2019-07-27 09:13:13 -04:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                 self.bosses[str(player)]["Ganons Tower Basement"] = \ | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     self.multiworld.get_dungeon('Inverted Ganons Tower', player).bosses['bottom'].name | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                 self.bosses[str(player)]["Ganons Tower Middle"] = \ | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     self.multiworld.get_dungeon('Inverted Ganons Tower', player).bosses['middle'].name | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                 self.bosses[str(player)]["Ganons Tower Top"] = \ | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     self.multiworld.get_dungeon('Inverted Ganons Tower', player).bosses['top'].name | 
					
						
							| 
									
										
										
										
											2019-07-27 09:13:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |             self.bosses[str(player)]["Ganons Tower"] = "Agahnim 2" | 
					
						
							|  |  |  |             self.bosses[str(player)]["Ganon"] = "Ganon" | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-11 20:48:26 +01:00
										 |  |  |     def create_playthrough(self, create_paths: bool = True): | 
					
						
							|  |  |  |         """Destructive to the world while it is run, damage gets repaired afterwards.""" | 
					
						
							|  |  |  |         from itertools import chain | 
					
						
							|  |  |  |         # get locations containing progress items | 
					
						
							|  |  |  |         multiworld = self.multiworld | 
					
						
							|  |  |  |         prog_locations = {location for location in multiworld.get_filled_locations() if location.item.advancement} | 
					
						
							|  |  |  |         state_cache = [None] | 
					
						
							|  |  |  |         collection_spheres: List[Set[Location]] = [] | 
					
						
							|  |  |  |         state = CollectionState(multiworld) | 
					
						
							|  |  |  |         sphere_candidates = set(prog_locations) | 
					
						
							|  |  |  |         logging.debug('Building up collection spheres.') | 
					
						
							|  |  |  |         while sphere_candidates: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # build up spheres of collection radius. | 
					
						
							|  |  |  |             # Everything in each sphere is independent from each other in dependencies and only depends on lower spheres | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             sphere = {location for location in sphere_candidates if state.can_reach(location)} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for location in sphere: | 
					
						
							|  |  |  |                 state.collect(location.item, True, location) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             sphere_candidates -= sphere | 
					
						
							|  |  |  |             collection_spheres.append(sphere) | 
					
						
							|  |  |  |             state_cache.append(state.copy()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             logging.debug('Calculated sphere %i, containing %i of %i progress items.', len(collection_spheres), | 
					
						
							|  |  |  |                           len(sphere), | 
					
						
							|  |  |  |                           len(prog_locations)) | 
					
						
							|  |  |  |             if not sphere: | 
					
						
							|  |  |  |                 logging.debug('The following items could not be reached: %s', ['%s (Player %d) at %s (Player %d)' % ( | 
					
						
							|  |  |  |                     location.item.name, location.item.player, location.name, location.player) for location in | 
					
						
							|  |  |  |                                                                                sphere_candidates]) | 
					
						
							|  |  |  |                 if any([multiworld.accessibility[location.item.player] != 'minimal' for location in sphere_candidates]): | 
					
						
							|  |  |  |                     raise RuntimeError(f'Not all progression items reachable ({sphere_candidates}). ' | 
					
						
							|  |  |  |                                        f'Something went terribly wrong here.') | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     self.unreachables = sphere_candidates | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # in the second phase, we cull each sphere such that the game is still beatable, | 
					
						
							|  |  |  |         # reducing each range of influence to the bare minimum required inside it | 
					
						
							|  |  |  |         restore_later = {} | 
					
						
							|  |  |  |         for num, sphere in reversed(tuple(enumerate(collection_spheres))): | 
					
						
							|  |  |  |             to_delete = set() | 
					
						
							|  |  |  |             for location in sphere: | 
					
						
							|  |  |  |                 # we remove the item at location and check if game is still beatable | 
					
						
							|  |  |  |                 logging.debug('Checking if %s (Player %d) is required to beat the game.', location.item.name, | 
					
						
							|  |  |  |                               location.item.player) | 
					
						
							|  |  |  |                 old_item = location.item | 
					
						
							|  |  |  |                 location.item = None | 
					
						
							|  |  |  |                 if multiworld.can_beat_game(state_cache[num]): | 
					
						
							|  |  |  |                     to_delete.add(location) | 
					
						
							|  |  |  |                     restore_later[location] = old_item | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     # still required, got to keep it around | 
					
						
							|  |  |  |                     location.item = old_item | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # cull entries in spheres for spoiler walkthrough at end | 
					
						
							|  |  |  |             sphere -= to_delete | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # second phase, sphere 0 | 
					
						
							|  |  |  |         removed_precollected = [] | 
					
						
							|  |  |  |         for item in (i for i in chain.from_iterable(multiworld.precollected_items.values()) if i.advancement): | 
					
						
							|  |  |  |             logging.debug('Checking if %s (Player %d) is required to beat the game.', item.name, item.player) | 
					
						
							|  |  |  |             multiworld.precollected_items[item.player].remove(item) | 
					
						
							|  |  |  |             multiworld.state.remove(item) | 
					
						
							|  |  |  |             if not multiworld.can_beat_game(): | 
					
						
							|  |  |  |                 multiworld.push_precollected(item) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 removed_precollected.append(item) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # we are now down to just the required progress items in collection_spheres. Unfortunately | 
					
						
							|  |  |  |         # the previous pruning stage could potentially have made certain items dependant on others | 
					
						
							|  |  |  |         # in the same or later sphere (because the location had 2 ways to access but the item originally | 
					
						
							|  |  |  |         # used to access it was deemed not required.) So we need to do one final sphere collection pass | 
					
						
							|  |  |  |         # to build up the correct spheres | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         required_locations = {item for sphere in collection_spheres for item in sphere} | 
					
						
							|  |  |  |         state = CollectionState(multiworld) | 
					
						
							|  |  |  |         collection_spheres = [] | 
					
						
							|  |  |  |         while required_locations: | 
					
						
							|  |  |  |             state.sweep_for_events(key_only=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             sphere = set(filter(state.can_reach, required_locations)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for location in sphere: | 
					
						
							|  |  |  |                 state.collect(location.item, True, location) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             required_locations -= sphere | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             collection_spheres.append(sphere) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             logging.debug('Calculated final sphere %i, containing %i of %i progress items.', len(collection_spheres), | 
					
						
							|  |  |  |                           len(sphere), len(required_locations)) | 
					
						
							|  |  |  |             if not sphere: | 
					
						
							|  |  |  |                 raise RuntimeError(f'Not all required items reachable. Unreachable locations: {required_locations}') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # we can finally output our playthrough | 
					
						
							|  |  |  |         self.playthrough = {"0": sorted([str(item) for item in | 
					
						
							|  |  |  |                                          chain.from_iterable(multiworld.precollected_items.values()) | 
					
						
							|  |  |  |                                          if item.advancement])} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for i, sphere in enumerate(collection_spheres): | 
					
						
							|  |  |  |             self.playthrough[str(i + 1)] = { | 
					
						
							|  |  |  |                 str(location): str(location.item) for location in sorted(sphere)} | 
					
						
							|  |  |  |         if create_paths: | 
					
						
							|  |  |  |             self.create_paths(state, collection_spheres) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # repair the multiworld again | 
					
						
							|  |  |  |         for location, item in restore_later.items(): | 
					
						
							|  |  |  |             location.item = item | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for item in removed_precollected: | 
					
						
							|  |  |  |             multiworld.push_precollected(item) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def create_paths(self, state: CollectionState, collection_spheres: List[Set[Location]]): | 
					
						
							|  |  |  |         from itertools import zip_longest | 
					
						
							|  |  |  |         multiworld = self.multiworld | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def flist_to_iter(node): | 
					
						
							|  |  |  |             while node: | 
					
						
							|  |  |  |                 value, node = node | 
					
						
							|  |  |  |                 yield value | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def get_path(state, region): | 
					
						
							|  |  |  |             reversed_path_as_flist = state.path.get(region, (region, None)) | 
					
						
							|  |  |  |             string_path_flat = reversed(list(map(str, flist_to_iter(reversed_path_as_flist)))) | 
					
						
							|  |  |  |             # Now we combine the flat string list into (region, exit) pairs | 
					
						
							|  |  |  |             pathsiter = iter(string_path_flat) | 
					
						
							|  |  |  |             pathpairs = zip_longest(pathsiter, pathsiter) | 
					
						
							|  |  |  |             return list(pathpairs) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.paths = {} | 
					
						
							|  |  |  |         topology_worlds = (player for player in multiworld.player_ids if multiworld.worlds[player].topology_present) | 
					
						
							|  |  |  |         for player in topology_worlds: | 
					
						
							|  |  |  |             self.paths.update( | 
					
						
							|  |  |  |                 {str(location): get_path(state, location.parent_region) | 
					
						
							|  |  |  |                  for sphere in collection_spheres for location in sphere | 
					
						
							|  |  |  |                  if location.player == player}) | 
					
						
							|  |  |  |             if player in multiworld.get_game_players("A Link to the Past"): | 
					
						
							|  |  |  |                 # If Pyramid Fairy Entrance needs to be reached, also path to Big Bomb Shop | 
					
						
							|  |  |  |                 # Maybe move the big bomb over to the Event system instead? | 
					
						
							|  |  |  |                 if any(exit_path == 'Pyramid Fairy' for path in self.paths.values() | 
					
						
							|  |  |  |                        for (_, exit_path) in path): | 
					
						
							|  |  |  |                     if multiworld.mode[player] != 'inverted': | 
					
						
							|  |  |  |                         self.paths[str(multiworld.get_region('Big Bomb Shop', player))] = \ | 
					
						
							|  |  |  |                             get_path(state, multiworld.get_region('Big Bomb Shop', player)) | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         self.paths[str(multiworld.get_region('Inverted Big Bomb Shop', player))] = \ | 
					
						
							|  |  |  |                             get_path(state, multiworld.get_region('Inverted Big Bomb Shop', player)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  |     def to_json(self): | 
					
						
							|  |  |  |         self.parse_data() | 
					
						
							|  |  |  |         out = OrderedDict() | 
					
						
							| 
									
										
										
										
											2018-03-26 21:39:48 -04:00
										 |  |  |         out['Entrances'] = list(self.entrances.values()) | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  |         out.update(self.locations) | 
					
						
							| 
									
										
										
										
											2018-03-26 21:39:48 -04:00
										 |  |  |         out['Special'] = self.medallions | 
					
						
							| 
									
										
										
										
											2020-01-14 10:42:27 +01:00
										 |  |  |         if self.hashes: | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  |             out['Hashes'] = self.hashes | 
					
						
							| 
									
										
										
										
											2018-03-26 21:39:48 -04:00
										 |  |  |         if self.shops: | 
					
						
							|  |  |  |             out['Shops'] = self.shops | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  |         out['playthrough'] = self.playthrough | 
					
						
							| 
									
										
										
										
											2018-01-01 15:55:13 -05:00
										 |  |  |         out['paths'] = self.paths | 
					
						
							| 
									
										
										
										
											2019-12-17 15:55:53 +01:00
										 |  |  |         out['Bosses'] = self.bosses | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  |         return json.dumps(out) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  |     def to_file(self, filename: str): | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  |         self.parse_data() | 
					
						
							| 
									
										
										
										
											2020-08-19 23:24:17 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-20 20:13:00 +02:00
										 |  |  |         def bool_to_text(variable: Union[bool, str]) -> str: | 
					
						
							|  |  |  |             if type(variable) == str: | 
					
						
							|  |  |  |                 return variable | 
					
						
							| 
									
										
										
										
											2020-08-19 23:24:17 +02:00
										 |  |  |             return 'Yes' if variable else 'No' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  |         def write_option(option_key: str, option_obj: type(Options.Option)): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             res = getattr(self.multiworld, option_key)[player] | 
					
						
							| 
									
										
										
										
											2022-02-02 16:29:29 +01:00
										 |  |  |             display_name = getattr(option_obj, "display_name", option_key) | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2022-02-02 16:29:29 +01:00
										 |  |  |                 outfile.write(f'{display_name + ":":33}{res.get_current_option_name()}\n') | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  |             except: | 
					
						
							|  |  |  |                 raise Exception | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-10 00:36:26 +01:00
										 |  |  |         with open(filename, 'w', encoding="utf-8-sig") as outfile: | 
					
						
							|  |  |  |             outfile.write( | 
					
						
							| 
									
										
										
										
											2021-01-03 14:32:32 +01:00
										 |  |  |                 'Archipelago Version %s  -  Seed: %s\n\n' % ( | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     Utils.__version__, self.multiworld.seed)) | 
					
						
							|  |  |  |             outfile.write('Filling Algorithm:               %s\n' % self.multiworld.algorithm) | 
					
						
							|  |  |  |             outfile.write('Players:                         %d\n' % self.multiworld.players) | 
					
						
							| 
									
										
										
										
											2023-01-17 17:25:59 +01:00
										 |  |  |             outfile.write(f'Plando Options:                  {self.multiworld.plando_options}\n') | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             AutoWorld.call_stage(self.multiworld, "write_spoiler_header", outfile) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for player in range(1, self.multiworld.players + 1): | 
					
						
							|  |  |  |                 if self.multiworld.players > 1: | 
					
						
							|  |  |  |                     outfile.write('\nPlayer %d: %s\n' % (player, self.multiworld.get_player_name(player))) | 
					
						
							|  |  |  |                 outfile.write('Game:                            %s\n' % self.multiworld.game[player]) | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  |                 for f_option, option in Options.per_game_common_options.items(): | 
					
						
							|  |  |  |                     write_option(f_option, option) | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 options = self.multiworld.worlds[player].option_definitions | 
					
						
							| 
									
										
										
										
											2021-06-25 23:32:13 +02:00
										 |  |  |                 if options: | 
					
						
							| 
									
										
										
										
											2021-08-02 04:57:57 +02:00
										 |  |  |                     for f_option, option in options.items(): | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  |                         write_option(f_option, option) | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 AutoWorld.call_single(self.multiworld, "write_spoiler_header", player, outfile) | 
					
						
							| 
									
										
										
										
											2021-06-04 00:29:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 if player in self.multiworld.get_game_players("A Link to the Past"): | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  |                     outfile.write('%s%s\n' % ('Hash: ', self.hashes[player])) | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     outfile.write('Logic:                           %s\n' % self.multiworld.logic[player]) | 
					
						
							|  |  |  |                     outfile.write('Dark Room Logic:                 %s\n' % self.multiworld.dark_room_logic[player]) | 
					
						
							|  |  |  |                     outfile.write('Mode:                            %s\n' % self.multiworld.mode[player]) | 
					
						
							|  |  |  |                     outfile.write('Goal:                            %s\n' % self.multiworld.goal[player]) | 
					
						
							|  |  |  |                     if "triforce" in self.multiworld.goal[player]:  # triforce hunt | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |                         outfile.write("Pieces available for Triforce:   %s\n" % | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                                       self.multiworld.triforce_pieces_available[player]) | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |                         outfile.write("Pieces required for Triforce:    %s\n" % | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                                       self.multiworld.triforce_pieces_required[player]) | 
					
						
							|  |  |  |                     outfile.write('Difficulty:                      %s\n' % self.multiworld.difficulty[player]) | 
					
						
							|  |  |  |                     outfile.write('Item Functionality:              %s\n' % self.multiworld.item_functionality[player]) | 
					
						
							|  |  |  |                     outfile.write('Entrance Shuffle:                %s\n' % self.multiworld.shuffle[player]) | 
					
						
							|  |  |  |                     if self.multiworld.shuffle[player] != "vanilla": | 
					
						
							|  |  |  |                         outfile.write('Entrance Shuffle Seed            %s\n' % self.multiworld.worlds[player].er_seed) | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |                     outfile.write('Shop inventory shuffle:          %s\n' % | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                                   bool_to_text("i" in self.multiworld.shop_shuffle[player])) | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |                     outfile.write('Shop price shuffle:              %s\n' % | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                                   bool_to_text("p" in self.multiworld.shop_shuffle[player])) | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |                     outfile.write('Shop upgrade shuffle:            %s\n' % | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                                   bool_to_text("u" in self.multiworld.shop_shuffle[player])) | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |                     outfile.write('New Shop inventory:              %s\n' % | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                                   bool_to_text("g" in self.multiworld.shop_shuffle[player] or | 
					
						
							|  |  |  |                                                "f" in self.multiworld.shop_shuffle[player])) | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |                     outfile.write('Custom Potion Shop:              %s\n' % | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                                   bool_to_text("w" in self.multiworld.shop_shuffle[player])) | 
					
						
							|  |  |  |                     outfile.write('Enemy health:                    %s\n' % self.multiworld.enemy_health[player]) | 
					
						
							|  |  |  |                     outfile.write('Enemy damage:                    %s\n' % self.multiworld.enemy_damage[player]) | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |                     outfile.write('Prize shuffle                    %s\n' % | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                                   self.multiworld.shuffle_prizes[player]) | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  |             if self.entrances: | 
					
						
							|  |  |  |                 outfile.write('\n\nEntrances:\n\n') | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 outfile.write('\n'.join(['%s%s %s %s' % (f'{self.multiworld.get_player_name(entry["player"])}: ' | 
					
						
							|  |  |  |                                                          if self.multiworld.players > 1 else '', entry['entrance'], | 
					
						
							| 
									
										
										
										
											2020-08-23 21:38:21 +02:00
										 |  |  |                                                          '<=>' if entry['direction'] == 'both' else | 
					
						
							|  |  |  |                                                          '<=' if entry['direction'] == 'exit' else '=>', | 
					
						
							|  |  |  |                                                          entry['exit']) for entry in self.entrances.values()])) | 
					
						
							| 
									
										
										
										
											2021-06-06 17:13:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if self.medallions: | 
					
						
							| 
									
										
										
										
											2021-05-22 09:42:15 -05:00
										 |  |  |                 outfile.write('\n\nMedallions:\n') | 
					
						
							|  |  |  |                 for dungeon, medallion in self.medallions.items(): | 
					
						
							|  |  |  |                     outfile.write(f'\n{dungeon}: {medallion}') | 
					
						
							| 
									
										
										
										
											2021-11-02 12:29:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             AutoWorld.call_all(self.multiworld, "write_spoiler", outfile) | 
					
						
							| 
									
										
										
										
											2021-07-07 10:14:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  |             outfile.write('\n\nLocations:\n\n') | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |             outfile.write('\n'.join( | 
					
						
							|  |  |  |                 ['%s: %s' % (location, item) for grouping in self.locations.values() for (location, item) in | 
					
						
							|  |  |  |                  grouping.items()])) | 
					
						
							| 
									
										
										
										
											2021-06-06 17:13:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if self.shops: | 
					
						
							|  |  |  |                 outfile.write('\n\nShops:\n\n') | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                 outfile.write('\n'.join("{} [{}]\n    {}".format(shop['location'], shop['type'], "\n    ".join( | 
					
						
							|  |  |  |                     item for item in [shop.get('item_0', None), shop.get('item_1', None), shop.get('item_2', None)] if | 
					
						
							|  |  |  |                     item)) for shop in self.shops)) | 
					
						
							| 
									
										
										
										
											2021-06-06 17:13:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             for player in self.multiworld.get_game_players("A Link to the Past"): | 
					
						
							|  |  |  |                 if self.multiworld.boss_shuffle[player] != 'none': | 
					
						
							|  |  |  |                     bossmap = self.bosses[str(player)] if self.multiworld.players > 1 else self.bosses | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                     outfile.write( | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                         f'\n\nBosses{(f" ({self.multiworld.get_player_name(player)})" if self.multiworld.players > 1 else "")}:\n') | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                     outfile.write('    ' + '\n    '.join([f'{x}: {y}' for x, y in bossmap.items()])) | 
					
						
							| 
									
										
										
										
											2017-07-18 12:44:13 +02:00
										 |  |  |             outfile.write('\n\nPlaythrough:\n\n') | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |             outfile.write('\n'.join(['%s: {\n%s\n}' % (sphere_nr, '\n'.join( | 
					
						
							|  |  |  |                 ['  %s: %s' % (location, item) for (location, item) in sphere.items()] if sphere_nr != '0' else [ | 
					
						
							|  |  |  |                     f'  {item}' for item in sphere])) for (sphere_nr, sphere) in self.playthrough.items()])) | 
					
						
							| 
									
										
										
										
											2019-12-21 13:33:07 +01:00
										 |  |  |             if self.unreachables: | 
					
						
							|  |  |  |                 outfile.write('\n\nUnreachable Items:\n\n') | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                 outfile.write( | 
					
						
							|  |  |  |                     '\n'.join(['%s: %s' % (unreachable.item, unreachable) for unreachable in self.unreachables])) | 
					
						
							| 
									
										
										
										
											2018-01-01 15:55:13 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-06 17:13:34 +02:00
										 |  |  |             if self.paths: | 
					
						
							|  |  |  |                 outfile.write('\n\nPaths:\n\n') | 
					
						
							|  |  |  |                 path_listings = [] | 
					
						
							|  |  |  |                 for location, path in sorted(self.paths.items()): | 
					
						
							|  |  |  |                     path_lines = [] | 
					
						
							|  |  |  |                     for region, exit in path: | 
					
						
							|  |  |  |                         if exit is not None: | 
					
						
							|  |  |  |                             path_lines.append("{} -> {}".format(region, exit)) | 
					
						
							|  |  |  |                         else: | 
					
						
							|  |  |  |                             path_lines.append(region) | 
					
						
							|  |  |  |                     path_listings.append("{}\n        {}".format(location, "\n   =>   ".join(path_lines))) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 outfile.write('\n'.join(path_listings)) | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             AutoWorld.call_all(self.multiworld, "write_spoiler_end", outfile) | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-11 13:05:53 -05:00
										 |  |  | class Tutorial(NamedTuple): | 
					
						
							|  |  |  |     """Class to build website tutorial pages from a .md file in the world's /docs folder. Order is as follows.
 | 
					
						
							|  |  |  |     Name of the tutorial as it will appear on the site. Concise description covering what the guide will entail. | 
					
						
							|  |  |  |     Language the guide is written in. Name of the file ex 'setup_en.md'. Name of the link on the site; game name is | 
					
						
							|  |  |  |     filled automatically so 'setup/en' etc. Author or authors."""
 | 
					
						
							|  |  |  |     tutorial_name: str | 
					
						
							|  |  |  |     description: str | 
					
						
							|  |  |  |     language: str | 
					
						
							|  |  |  |     file_name: str | 
					
						
							|  |  |  |     link: str | 
					
						
							| 
									
										
										
										
											2022-05-26 20:39:08 -04:00
										 |  |  |     authors: List[str] | 
					
						
							| 
									
										
										
										
											2022-05-11 13:05:53 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-17 17:25:59 +01:00
										 |  |  | class PlandoOptions(IntFlag): | 
					
						
							| 
									
										
										
										
											2023-01-15 11:10:26 -06:00
										 |  |  |     none = 0b0000 | 
					
						
							|  |  |  |     items = 0b0001 | 
					
						
							|  |  |  |     connections = 0b0010 | 
					
						
							|  |  |  |     texts = 0b0100 | 
					
						
							|  |  |  |     bosses = 0b1000 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							| 
									
										
										
										
											2023-01-17 17:25:59 +01:00
										 |  |  |     def from_option_string(cls, option_string: str) -> PlandoOptions: | 
					
						
							| 
									
										
										
										
											2023-01-15 11:10:26 -06:00
										 |  |  |         result = cls(0) | 
					
						
							|  |  |  |         for part in option_string.split(","): | 
					
						
							|  |  |  |             part = part.strip().lower() | 
					
						
							|  |  |  |             if part: | 
					
						
							|  |  |  |                 result = cls._handle_part(part, result) | 
					
						
							|  |  |  |         return result | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							| 
									
										
										
										
											2023-01-17 17:25:59 +01:00
										 |  |  |     def from_set(cls, option_set: Set[str]) -> PlandoOptions: | 
					
						
							| 
									
										
										
										
											2023-01-15 11:10:26 -06:00
										 |  |  |         result = cls(0) | 
					
						
							|  |  |  |         for part in option_set: | 
					
						
							|  |  |  |             result = cls._handle_part(part, result) | 
					
						
							|  |  |  |         return result | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							| 
									
										
										
										
											2023-01-17 17:25:59 +01:00
										 |  |  |     def _handle_part(cls, part: str, base: PlandoOptions) -> PlandoOptions: | 
					
						
							| 
									
										
										
										
											2023-01-15 11:10:26 -06:00
										 |  |  |         try: | 
					
						
							|  |  |  |             part = cls[part] | 
					
						
							|  |  |  |         except Exception as e: | 
					
						
							|  |  |  |             raise KeyError(f"{part} is not a recognized name for a plando module. " | 
					
						
							|  |  |  |                            f"Known options: {', '.join(flag.name for flag in cls)}") from e | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return base | part | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __str__(self) -> str: | 
					
						
							|  |  |  |         if self.value: | 
					
						
							| 
									
										
										
										
											2023-01-17 17:25:59 +01:00
										 |  |  |             return ", ".join(flag.name for flag in PlandoOptions if self.value & flag.value) | 
					
						
							| 
									
										
										
										
											2023-01-15 11:10:26 -06:00
										 |  |  |         return "None" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  | seeddigits = 20 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-24 00:46:26 +01:00
										 |  |  | def get_seed(seed=None) -> int: | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |     if seed is None: | 
					
						
							|  |  |  |         random.seed(None) | 
					
						
							|  |  |  |         return random.randint(0, pow(10, seeddigits) - 1) | 
					
						
							|  |  |  |     return seed | 
					
						
							| 
									
										
										
										
											2022-02-18 20:29:35 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from worlds import AutoWorld | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-11 13:05:53 -05:00
										 |  |  | auto_world = AutoWorld.World |