| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  | """
 | 
					
						
							|  |  |  | Defines Region for The Witness, assigns locations to them, | 
					
						
							|  |  |  | and connects them with the proper requirements | 
					
						
							|  |  |  | """
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  | from typing import FrozenSet, TYPE_CHECKING, Dict, Tuple, List | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  | from BaseClasses import Entrance, Region | 
					
						
							|  |  |  | from Utils import KeyedDefaultDict | 
					
						
							| 
									
										
										
										
											2022-10-09 04:13:52 +02:00
										 |  |  | from .static_logic import StaticWitnessLogic | 
					
						
							| 
									
										
										
										
											2023-02-01 21:18:07 +01:00
										 |  |  | from .locations import WitnessPlayerLocations, StaticWitnessLocations | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  | from .player_logic import WitnessPlayerLogic | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  | if TYPE_CHECKING: | 
					
						
							|  |  |  |     from . import WitnessWorld | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | class WitnessRegions: | 
					
						
							|  |  |  |     """Class that defines Witness Regions""" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     locat = None | 
					
						
							|  |  |  |     logic = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def make_lambda(item_requirement: FrozenSet[FrozenSet[str]], world: "WitnessWorld"): | 
					
						
							|  |  |  |         from .rules import _meets_item_requirements | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Lambdas are made in a for loop, so the values have to be captured | 
					
						
							|  |  |  |         This function is for that purpose | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |         return _meets_item_requirements(item_requirement, world) | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |     def connect_if_possible(self, world: "WitnessWorld", source: str, target: str, req: FrozenSet[FrozenSet[str]], | 
					
						
							|  |  |  |                             regions_by_name: Dict[str, Region], backwards: bool = False): | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         connect two regions and set the corresponding requirement | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Remove any possibilities where being in the target region would be required anyway. | 
					
						
							|  |  |  |         real_requirement = frozenset({option for option in req if target not in option}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # There are some connections that should only be done one way. If this is a backwards connection, check for that | 
					
						
							|  |  |  |         if backwards: | 
					
						
							|  |  |  |             real_requirement = frozenset({option for option in real_requirement if "TrueOneWay" not in option}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Dissolve any "True" or "TrueOneWay" | 
					
						
							|  |  |  |         real_requirement = frozenset({option - {"True", "TrueOneWay"} for option in real_requirement}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # If there is no way to actually use this connection, don't even bother making it. | 
					
						
							|  |  |  |         if not real_requirement: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # We don't need to check for the accessibility of the source region. | 
					
						
							|  |  |  |         final_requirement = frozenset({option - frozenset({source}) for option in real_requirement}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         source_region = regions_by_name[source] | 
					
						
							|  |  |  |         target_region = regions_by_name[target] | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-03 00:08:24 +01:00
										 |  |  |         backwards = " Backwards" if backwards else "" | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |         connection_name = source + " to " + target + backwards | 
					
						
							| 
									
										
										
										
											2022-07-17 12:56:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  |         connection = Entrance( | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |             world.player, | 
					
						
							|  |  |  |             connection_name, | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  |             source_region | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |         connection.access_rule = self.make_lambda(final_requirement, world) | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         source_region.exits.append(connection) | 
					
						
							|  |  |  |         connection.connect(target_region) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-07 06:36:46 +01:00
										 |  |  |         self.created_entrances[source, target].append(connection) | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Register any necessary indirect connections | 
					
						
							|  |  |  |         mentioned_regions = { | 
					
						
							|  |  |  |             single_unlock for option in final_requirement for single_unlock in option | 
					
						
							|  |  |  |             if single_unlock in self.reference_logic.ALL_REGIONS_BY_NAME | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for dependent_region in mentioned_regions: | 
					
						
							|  |  |  |             world.multiworld.register_indirect_condition(regions_by_name[dependent_region], connection) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def create_regions(self, world: "WitnessWorld", player_logic: WitnessPlayerLogic): | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Creates all the regions for The Witness | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         from . import create_region | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         all_locations = set() | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |         regions_by_name = dict() | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |         for region_name, region in self.reference_logic.ALL_REGIONS_BY_NAME.items(): | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  |             locations_for_this_region = [ | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |                 self.reference_logic.ENTITIES_BY_HEX[panel]["checkName"] for panel in region["panels"] | 
					
						
							|  |  |  |                 if self.reference_logic.ENTITIES_BY_HEX[panel]["checkName"] in self.locat.CHECK_LOCATION_TABLE | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  |             ] | 
					
						
							|  |  |  |             locations_for_this_region += [ | 
					
						
							| 
									
										
										
										
											2023-02-01 21:18:07 +01:00
										 |  |  |                 StaticWitnessLocations.get_event_name(panel) for panel in region["panels"] | 
					
						
							|  |  |  |                 if StaticWitnessLocations.get_event_name(panel) in self.locat.EVENT_LOCATION_TABLE | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  |             ] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             all_locations = all_locations | set(locations_for_this_region) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |             new_region = create_region(world, region_name, self.locat, locations_for_this_region) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             regions_by_name[region_name] = new_region | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |         for region_name, region in self.reference_logic.ALL_REGIONS_BY_NAME.items(): | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  |             for connection in player_logic.CONNECTIONS_BY_REGION_NAME[region_name]: | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |                 self.connect_if_possible(world, region_name, connection[0], connection[1], regions_by_name) | 
					
						
							|  |  |  |                 self.connect_if_possible(world, connection[0], region_name, connection[1], regions_by_name, True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # find regions that are completely disconnected from the start node and remove them | 
					
						
							|  |  |  |         regions_to_check = {"Menu"} | 
					
						
							|  |  |  |         reachable_regions = {"Menu"} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while regions_to_check: | 
					
						
							|  |  |  |             next_region = regions_to_check.pop() | 
					
						
							|  |  |  |             region_obj = regions_by_name[next_region] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for exit in region_obj.exits: | 
					
						
							|  |  |  |                 target = exit.connected_region | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if target.name in reachable_regions: | 
					
						
							| 
									
										
										
										
											2022-07-17 12:56:22 +02:00
										 |  |  |                     continue | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |                 regions_to_check.add(target.name) | 
					
						
							|  |  |  |                 reachable_regions.add(target.name) | 
					
						
							| 
									
										
										
										
											2023-03-03 00:08:24 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |         final_regions_list = [v for k, v in regions_by_name.items() if k in reachable_regions] | 
					
						
							| 
									
										
										
										
											2023-03-03 00:08:24 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |         world.multiworld.regions += final_regions_list | 
					
						
							| 
									
										
										
										
											2022-07-17 12:56:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |     def __init__(self, locat: WitnessPlayerLocations, world: "WitnessWorld"): | 
					
						
							|  |  |  |         difficulty = world.options.puzzle_randomization.value | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |         if difficulty == 0: | 
					
						
							|  |  |  |             self.reference_logic = StaticWitnessLogic.sigma_normal | 
					
						
							|  |  |  |         elif difficulty == 1: | 
					
						
							|  |  |  |             self.reference_logic = StaticWitnessLogic.sigma_expert | 
					
						
							|  |  |  |         elif difficulty == 2: | 
					
						
							|  |  |  |             self.reference_logic = StaticWitnessLogic.vanilla | 
					
						
							| 
									
										
										
										
											2022-04-29 00:42:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.locat = locat | 
					
						
							| 
									
										
										
										
											2023-11-24 06:27:03 +01:00
										 |  |  |         self.created_entrances: Dict[Tuple[str, str], List[Entrance]] = KeyedDefaultDict(lambda _: []) |