summary refs log tree commit diff stats
path: root/regions.py
diff options
context:
space:
mode:
authorStar Rauchenberger <fefferburbia@gmail.com>2024-04-18 11:45:33 -0500
committerGitHub <noreply@github.com>2024-04-18 18:45:33 +0200
commitb45b9bd74612239ebc29322fc340b1bee62e7032 (patch)
treed61b0077586fa146a418a46536daef02ae27989a /regions.py
parent7a358cedc44c0892a4c369a4884a23a001535d63 (diff)
downloadlingo-apworld-b45b9bd74612239ebc29322fc340b1bee62e7032.tar.gz
lingo-apworld-b45b9bd74612239ebc29322fc340b1bee62e7032.tar.bz2
lingo-apworld-b45b9bd74612239ebc29322fc340b1bee62e7032.zip
Lingo: The Pilgrim Update (#2884)
* An option was added to enable or disable the pilgrimage, and it defaults to disabled. When disabled, the client will prevent you from performing a pilgrimage (i.e. the yellow border will not appear when you enter the 1 sunwarp). The sun painting is added to the item pool when pilgrimage is disabled, as otherwise there is no way into the Pilgrim Antechamber. Inversely, the sun painting is no longer in the item pool when pilgrimage is enabled (even if door shuffle is on), requiring you to perform a pilgrimage to get to that room.
* The canonical pilgrimage has been deprecated. Instead, there is logic for determining whether a pilgrimage is possible.
* Two options were added that allow the player to decide whether paintings and/or Crossroads - Roof Access are permitted during the pilgrimage. Both default to disabled. These options apply both to logical expectations in the generator, and are also enforced by the game client.
* An option was added to control how sunwarps are accessed. The default is for them to always be accessible, like in the base game. It is also possible to disable them entirely (which is not possible when pilgrimage is enabled), or lock them behind items similar to door shuffle. It can either be one item that unlocks all sunwarps at the same time, six progressive items that unlock the sunwarps from 1 to 6, or six individual items that unlock the sunwarps in any order. This option is independent from door shuffle.
* An option was added that shuffles sunwarps. This acts similarly to painting shuffle. The 12 sunwarps are re-ordered and re-paired. Sunwarps that were previously entrances or exits do not need to stay entrances or exits. Performing a pilgrimage requires proceeding through the sunwarps in the new order, rather than the original one.
* Pilgrimage was added as a win condition. It requires you to solve the blue PILGRIM panel in the Pilgrim Antechamber.
Diffstat (limited to 'regions.py')
-rw-r--r--regions.py105
1 files changed, 94 insertions, 11 deletions
diff --git a/regions.py b/regions.py index 5fddabd..4b357db 100644 --- a/regions.py +++ b/regions.py
@@ -1,10 +1,11 @@
1from typing import Dict, Optional, TYPE_CHECKING 1from typing import Dict, Optional, TYPE_CHECKING
2 2
3from BaseClasses import Entrance, ItemClassification, Region 3from BaseClasses import Entrance, ItemClassification, Region
4from .datatypes import Room, RoomAndDoor 4from .datatypes import EntranceType, Room, RoomAndDoor
5from .items import LingoItem 5from .items import LingoItem
6from .locations import LingoLocation 6from .locations import LingoLocation
7from .rules import lingo_can_use_entrance, make_location_lambda 7from .options import SunwarpAccess
8from .rules import lingo_can_do_pilgrimage, lingo_can_use_entrance, make_location_lambda
8from .static_logic import ALL_ROOMS, PAINTINGS 9from .static_logic import ALL_ROOMS, PAINTINGS
9 10
10if TYPE_CHECKING: 11if TYPE_CHECKING:
@@ -25,8 +26,20 @@ def create_region(room: Room, world: "LingoWorld") -> Region:
25 return new_region 26 return new_region
26 27
27 28
29def is_acceptable_pilgrimage_entrance(entrance_type: EntranceType, world: "LingoWorld") -> bool:
30 allowed_entrance_types = EntranceType.NORMAL
31
32 if world.options.pilgrimage_allows_paintings:
33 allowed_entrance_types |= EntranceType.PAINTING
34
35 if world.options.pilgrimage_allows_roof_access:
36 allowed_entrance_types |= EntranceType.CROSSROADS_ROOF_ACCESS
37
38 return bool(entrance_type & allowed_entrance_types)
39
40
28def connect_entrance(regions: Dict[str, Region], source_region: Region, target_region: Region, description: str, 41def connect_entrance(regions: Dict[str, Region], source_region: Region, target_region: Region, description: str,
29 door: Optional[RoomAndDoor], world: "LingoWorld"): 42 door: Optional[RoomAndDoor], entrance_type: EntranceType, pilgrimage: bool, world: "LingoWorld"):
30 connection = Entrance(world.player, description, source_region) 43 connection = Entrance(world.player, description, source_region)
31 connection.access_rule = lambda state: lingo_can_use_entrance(state, target_region.name, door, world) 44 connection.access_rule = lambda state: lingo_can_use_entrance(state, target_region.name, door, world)
32 45
@@ -38,6 +51,21 @@ def connect_entrance(regions: Dict[str, Region], source_region: Region, target_r
38 if door.door not in world.player_logic.item_by_door.get(effective_room, {}): 51 if door.door not in world.player_logic.item_by_door.get(effective_room, {}):
39 for region in world.player_logic.calculate_door_requirements(effective_room, door.door, world).rooms: 52 for region in world.player_logic.calculate_door_requirements(effective_room, door.door, world).rooms:
40 world.multiworld.register_indirect_condition(regions[region], connection) 53 world.multiworld.register_indirect_condition(regions[region], connection)
54
55 if not pilgrimage and world.options.enable_pilgrimage and is_acceptable_pilgrimage_entrance(entrance_type, world)\
56 and source_region.name != "Menu":
57 for part in range(1, 6):
58 pilgrimage_descriptor = f" (Pilgrimage Part {part})"
59 pilgrim_source_region = regions[f"{source_region.name}{pilgrimage_descriptor}"]
60 pilgrim_target_region = regions[f"{target_region.name}{pilgrimage_descriptor}"]
61
62 effective_door = door
63 if effective_door is not None:
64 effective_room = target_region.name if door.room is None else door.room
65 effective_door = RoomAndDoor(effective_room, door.door)
66
67 connect_entrance(regions, pilgrim_source_region, pilgrim_target_region,
68 f"{description}{pilgrimage_descriptor}", effective_door, entrance_type, True, world)
41 69
42 70
43def connect_painting(regions: Dict[str, Region], warp_enter: str, warp_exit: str, world: "LingoWorld") -> None: 71def connect_painting(regions: Dict[str, Region], warp_enter: str, warp_exit: str, world: "LingoWorld") -> None:
@@ -48,7 +76,8 @@ def connect_painting(regions: Dict[str, Region], warp_enter: str, warp_exit: str
48 source_region = regions[source_painting.room] 76 source_region = regions[source_painting.room]
49 77
50 entrance_name = f"{source_painting.room} to {target_painting.room} ({source_painting.id} Painting)" 78 entrance_name = f"{source_painting.room} to {target_painting.room} ({source_painting.id} Painting)"
51 connect_entrance(regions, source_region, target_region, entrance_name, source_painting.required_door, world) 79 connect_entrance(regions, source_region, target_region, entrance_name, source_painting.required_door,
80 EntranceType.PAINTING, False, world)
52 81
53 82
54def create_regions(world: "LingoWorld") -> None: 83def create_regions(world: "LingoWorld") -> None:
@@ -63,11 +92,26 @@ def create_regions(world: "LingoWorld") -> None:
63 for room in ALL_ROOMS: 92 for room in ALL_ROOMS:
64 regions[room.name] = create_region(room, world) 93 regions[room.name] = create_region(room, world)
65 94
95 if world.options.enable_pilgrimage:
96 for part in range(1, 6):
97 pilgrimage_region_name = f"{room.name} (Pilgrimage Part {part})"
98 regions[pilgrimage_region_name] = Region(pilgrimage_region_name, world.player, world.multiworld)
99
66 # Connect all created regions now that they exist. 100 # Connect all created regions now that they exist.
101 allowed_entrance_types = EntranceType.NORMAL | EntranceType.WARP | EntranceType.CROSSROADS_ROOF_ACCESS
102
103 if not painting_shuffle:
104 # Don't use the vanilla painting connections if we are shuffling paintings.
105 allowed_entrance_types |= EntranceType.PAINTING
106
107 if world.options.sunwarp_access != SunwarpAccess.option_disabled and not world.options.shuffle_sunwarps:
108 # Don't connect sunwarps if sunwarps are disabled or if we're shuffling sunwarps.
109 allowed_entrance_types |= EntranceType.SUNWARP
110
67 for room in ALL_ROOMS: 111 for room in ALL_ROOMS:
68 for entrance in room.entrances: 112 for entrance in room.entrances:
69 # Don't use the vanilla painting connections if we are shuffling paintings. 113 effective_entrance_type = entrance.type & allowed_entrance_types
70 if entrance.painting and painting_shuffle: 114 if not effective_entrance_type:
71 continue 115 continue
72 116
73 entrance_name = f"{entrance.room} to {room.name}" 117 entrance_name = f"{entrance.room} to {room.name}"
@@ -77,17 +121,56 @@ def create_regions(world: "LingoWorld") -> None:
77 else: 121 else:
78 entrance_name += f" (through {room.name} - {entrance.door.door})" 122 entrance_name += f" (through {room.name} - {entrance.door.door})"
79 123
80 connect_entrance(regions, regions[entrance.room], regions[room.name], entrance_name, entrance.door, world) 124 effective_door = entrance.door
125 if entrance.type == EntranceType.SUNWARP and world.options.sunwarp_access == SunwarpAccess.option_normal:
126 effective_door = None
127
128 connect_entrance(regions, regions[entrance.room], regions[room.name], entrance_name, effective_door,
129 effective_entrance_type, False, world)
130
131 if world.options.enable_pilgrimage:
132 # Connect the start of the pilgrimage. We check for all sunwarp items here.
133 pilgrim_start_from = regions[world.player_logic.sunwarp_entrances[0]]
134 pilgrim_start_to = regions[f"{world.player_logic.sunwarp_exits[0]} (Pilgrimage Part 1)"]
135
136 if world.options.sunwarp_access >= SunwarpAccess.option_unlock:
137 pilgrim_start_from.connect(pilgrim_start_to, f"Pilgrimage Part 1",
138 lambda state: lingo_can_do_pilgrimage(state, world))
139 else:
140 pilgrim_start_from.connect(pilgrim_start_to, f"Pilgrimage Part 1")
81 141
82 # Add the fake pilgrimage. 142 # Create connections between each segment of the pilgrimage.
83 connect_entrance(regions, regions["Outside The Agreeable"], regions["Pilgrim Antechamber"], "Pilgrimage", 143 for i in range(1, 6):
84 RoomAndDoor("Pilgrim Antechamber", "Pilgrimage"), world) 144 from_room = f"{world.player_logic.sunwarp_entrances[i]} (Pilgrimage Part {i})"
145 to_room = f"{world.player_logic.sunwarp_exits[i]} (Pilgrimage Part {i+1})"
146 if i == 5:
147 to_room = "Pilgrim Antechamber"
148
149 regions[from_room].connect(regions[to_room], f"Pilgrimage Part {i+1}")
150 else:
151 connect_entrance(regions, regions["Starting Room"], regions["Pilgrim Antechamber"], "Sun Painting",
152 RoomAndDoor("Pilgrim Antechamber", "Sun Painting"), EntranceType.PAINTING, False, world)
85 153
86 if early_color_hallways: 154 if early_color_hallways:
87 regions["Starting Room"].connect(regions["Outside The Undeterred"], "Early Color Hallways") 155 connect_entrance(regions, regions["Starting Room"], regions["Outside The Undeterred"], "Early Color Hallways",
156 None, EntranceType.PAINTING, False, world)
88 157
89 if painting_shuffle: 158 if painting_shuffle:
90 for warp_enter, warp_exit in world.player_logic.painting_mapping.items(): 159 for warp_enter, warp_exit in world.player_logic.painting_mapping.items():
91 connect_painting(regions, warp_enter, warp_exit, world) 160 connect_painting(regions, warp_enter, warp_exit, world)
92 161
162 if world.options.shuffle_sunwarps:
163 for i in range(0, 6):
164 if world.options.sunwarp_access == SunwarpAccess.option_normal:
165 effective_door = None
166 else:
167 effective_door = RoomAndDoor("Sunwarps", f"{i + 1} Sunwarp")
168
169 source_region = regions[world.player_logic.sunwarp_entrances[i]]
170 target_region = regions[world.player_logic.sunwarp_exits[i]]
171
172 entrance_name = f"{source_region.name} to {target_region.name} ({i + 1} Sunwarp)"
173 connect_entrance(regions, source_region, target_region, entrance_name, effective_door, EntranceType.SUNWARP,
174 False, world)
175
93 world.multiworld.regions += regions.values() 176 world.multiworld.regions += regions.values()