summary refs log tree commit diff stats
path: root/utils/pickle_static_data.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 /utils/pickle_static_data.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 'utils/pickle_static_data.py')
-rw-r--r--utils/pickle_static_data.py96
1 files changed, 63 insertions, 33 deletions
diff --git a/utils/pickle_static_data.py b/utils/pickle_static_data.py index 5d6fa1e..10ec69b 100644 --- a/utils/pickle_static_data.py +++ b/utils/pickle_static_data.py
@@ -1,4 +1,4 @@
1from typing import Dict, List, Set 1from typing import Dict, List, Set, Optional
2 2
3import os 3import os
4import sys 4import sys
@@ -6,7 +6,8 @@ import sys
6sys.path.append(os.path.join("worlds", "lingo")) 6sys.path.append(os.path.join("worlds", "lingo"))
7sys.path.append(".") 7sys.path.append(".")
8sys.path.append("..") 8sys.path.append("..")
9from datatypes import Door, Painting, Panel, Progression, Room, RoomAndDoor, RoomAndPanel, RoomEntrance 9from datatypes import Door, DoorType, EntranceType, Painting, Panel, Progression, Room, RoomAndDoor, RoomAndPanel,\
10 RoomEntrance
10 11
11import hashlib 12import hashlib
12import pickle 13import pickle
@@ -28,6 +29,9 @@ PAINTING_EXITS: int = 0
28REQUIRED_PAINTING_ROOMS: List[str] = [] 29REQUIRED_PAINTING_ROOMS: List[str] = []
29REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS: List[str] = [] 30REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS: List[str] = []
30 31
32SUNWARP_ENTRANCES: List[str] = ["", "", "", "", "", ""]
33SUNWARP_EXITS: List[str] = ["", "", "", "", "", ""]
34
31SPECIAL_ITEM_IDS: Dict[str, int] = {} 35SPECIAL_ITEM_IDS: Dict[str, int] = {}
32PANEL_LOCATION_IDS: Dict[str, Dict[str, int]] = {} 36PANEL_LOCATION_IDS: Dict[str, Dict[str, int]] = {}
33DOOR_LOCATION_IDS: Dict[str, Dict[str, int]] = {} 37DOOR_LOCATION_IDS: Dict[str, Dict[str, int]] = {}
@@ -96,41 +100,51 @@ def load_static_data(ll1_path, ids_path):
96 PAINTING_EXITS = len(PAINTING_EXIT_ROOMS) 100 PAINTING_EXITS = len(PAINTING_EXIT_ROOMS)
97 101
98 102
99def process_entrance(source_room, doors, room_obj): 103def process_single_entrance(source_room: str, room_name: str, door_obj) -> RoomEntrance:
100 global PAINTING_ENTRANCES, PAINTING_EXIT_ROOMS 104 global PAINTING_ENTRANCES, PAINTING_EXIT_ROOMS
101 105
106 entrance_type = EntranceType.NORMAL
107 if "painting" in door_obj and door_obj["painting"]:
108 entrance_type = EntranceType.PAINTING
109 elif "sunwarp" in door_obj and door_obj["sunwarp"]:
110 entrance_type = EntranceType.SUNWARP
111 elif "warp" in door_obj and door_obj["warp"]:
112 entrance_type = EntranceType.WARP
113 elif source_room == "Crossroads" and room_name == "Roof":
114 entrance_type = EntranceType.CROSSROADS_ROOF_ACCESS
115
116 if "painting" in door_obj and door_obj["painting"]:
117 PAINTING_EXIT_ROOMS.add(room_name)
118 PAINTING_ENTRANCES += 1
119
120 if "door" in door_obj:
121 return RoomEntrance(source_room, RoomAndDoor(
122 door_obj["room"] if "room" in door_obj else None,
123 door_obj["door"]
124 ), entrance_type)
125 else:
126 return RoomEntrance(source_room, None, entrance_type)
127
128
129def process_entrance(source_room, doors, room_obj):
102 # If the value of an entrance is just True, that means that the entrance is always accessible. 130 # If the value of an entrance is just True, that means that the entrance is always accessible.
103 if doors is True: 131 if doors is True:
104 room_obj.entrances.append(RoomEntrance(source_room, None, False)) 132 room_obj.entrances.append(RoomEntrance(source_room, None, EntranceType.NORMAL))
105 elif isinstance(doors, dict): 133 elif isinstance(doors, dict):
106 # If the value of an entrance is a dictionary, that means the entrance requires a door to be accessible, is a 134 # If the value of an entrance is a dictionary, that means the entrance requires a door to be accessible, is a
107 # painting-based entrance, or both. 135 # painting-based entrance, or both.
108 if "painting" in doors and "door" not in doors: 136 room_obj.entrances.append(process_single_entrance(source_room, room_obj.name, doors))
109 PAINTING_EXIT_ROOMS.add(room_obj.name)
110 PAINTING_ENTRANCES += 1
111
112 room_obj.entrances.append(RoomEntrance(source_room, None, True))
113 else:
114 if "painting" in doors and doors["painting"]:
115 PAINTING_EXIT_ROOMS.add(room_obj.name)
116 PAINTING_ENTRANCES += 1
117
118 room_obj.entrances.append(RoomEntrance(source_room, RoomAndDoor(
119 doors["room"] if "room" in doors else None,
120 doors["door"]
121 ), doors["painting"] if "painting" in doors else False))
122 else: 137 else:
123 # If the value of an entrance is a list, then there are multiple possible doors that can give access to the 138 # If the value of an entrance is a list, then there are multiple possible doors that can give access to the
124 # entrance. 139 # entrance. If there are multiple connections with the same door (or lack of door) that differ only by entrance
140 # type, coalesce them into one entrance.
141 entrances: Dict[Optional[RoomAndDoor], EntranceType] = {}
125 for door in doors: 142 for door in doors:
126 if "painting" in door and door["painting"]: 143 entrance = process_single_entrance(source_room, room_obj.name, door)
127 PAINTING_EXIT_ROOMS.add(room_obj.name) 144 entrances[entrance.door] = entrances.get(entrance.door, EntranceType(0)) | entrance.type
128 PAINTING_ENTRANCES += 1
129 145
130 room_obj.entrances.append(RoomEntrance(source_room, RoomAndDoor( 146 for door, entrance_type in entrances.items():
131 door["room"] if "room" in door else None, 147 room_obj.entrances.append(RoomEntrance(source_room, door, entrance_type))
132 door["door"]
133 ), door["painting"] if "painting" in door else False))
134 148
135 149
136def process_panel(room_name, panel_name, panel_data): 150def process_panel(room_name, panel_name, panel_data):
@@ -250,11 +264,6 @@ def process_door(room_name, door_name, door_data):
250 else: 264 else:
251 include_reduce = False 265 include_reduce = False
252 266
253 if "junk_item" in door_data:
254 junk_item = door_data["junk_item"]
255 else:
256 junk_item = False
257
258 if "door_group" in door_data: 267 if "door_group" in door_data:
259 door_group = door_data["door_group"] 268 door_group = door_data["door_group"]
260 else: 269 else:
@@ -276,7 +285,7 @@ def process_door(room_name, door_name, door_data):
276 panels.append(RoomAndPanel(None, panel)) 285 panels.append(RoomAndPanel(None, panel))
277 else: 286 else:
278 skip_location = True 287 skip_location = True
279 panels = None 288 panels = []
280 289
281 # The location name associated with a door can be explicitly specified in the configuration. If it is not, then the 290 # The location name associated with a door can be explicitly specified in the configuration. If it is not, then the
282 # name is generated using a combination of all of the panels that would ordinarily open the door. This can get quite 291 # name is generated using a combination of all of the panels that would ordinarily open the door. This can get quite
@@ -312,8 +321,14 @@ def process_door(room_name, door_name, door_data):
312 else: 321 else:
313 painting_ids = [] 322 painting_ids = []
314 323
324 door_type = DoorType.NORMAL
325 if door_name.endswith(" Sunwarp"):
326 door_type = DoorType.SUNWARP
327 elif room_name == "Pilgrim Antechamber" and door_name == "Sun Painting":
328 door_type = DoorType.SUN_PAINTING
329
315 door_obj = Door(door_name, item_name, location_name, panels, skip_location, skip_item, has_doors, 330 door_obj = Door(door_name, item_name, location_name, panels, skip_location, skip_item, has_doors,
316 painting_ids, event, door_group, include_reduce, junk_item, item_group) 331 painting_ids, event, door_group, include_reduce, door_type, item_group)
317 332
318 DOORS_BY_ROOM[room_name][door_name] = door_obj 333 DOORS_BY_ROOM[room_name][door_name] = door_obj
319 334
@@ -377,6 +392,15 @@ def process_painting(room_name, painting_data):
377 PAINTINGS[painting_id] = painting_obj 392 PAINTINGS[painting_id] = painting_obj
378 393
379 394
395def process_sunwarp(room_name, sunwarp_data):
396 global SUNWARP_ENTRANCES, SUNWARP_EXITS
397
398 if sunwarp_data["direction"] == "enter":
399 SUNWARP_ENTRANCES[sunwarp_data["dots"] - 1] = room_name
400 else:
401 SUNWARP_EXITS[sunwarp_data["dots"] - 1] = room_name
402
403
380def process_progression(room_name, progression_name, progression_doors): 404def process_progression(room_name, progression_name, progression_doors):
381 global PROGRESSIVE_ITEMS, PROGRESSION_BY_ROOM 405 global PROGRESSIVE_ITEMS, PROGRESSION_BY_ROOM
382 406
@@ -422,6 +446,10 @@ def process_room(room_name, room_data):
422 for painting_data in room_data["paintings"]: 446 for painting_data in room_data["paintings"]:
423 process_painting(room_name, painting_data) 447 process_painting(room_name, painting_data)
424 448
449 if "sunwarps" in room_data:
450 for sunwarp_data in room_data["sunwarps"]:
451 process_sunwarp(room_name, sunwarp_data)
452
425 if "progression" in room_data: 453 if "progression" in room_data:
426 for progression_name, progression_doors in room_data["progression"].items(): 454 for progression_name, progression_doors in room_data["progression"].items():
427 process_progression(room_name, progression_name, progression_doors) 455 process_progression(room_name, progression_name, progression_doors)
@@ -468,6 +496,8 @@ if __name__ == '__main__':
468 "PAINTING_EXITS": PAINTING_EXITS, 496 "PAINTING_EXITS": PAINTING_EXITS,
469 "REQUIRED_PAINTING_ROOMS": REQUIRED_PAINTING_ROOMS, 497 "REQUIRED_PAINTING_ROOMS": REQUIRED_PAINTING_ROOMS,
470 "REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS": REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS, 498 "REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS": REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS,
499 "SUNWARP_ENTRANCES": SUNWARP_ENTRANCES,
500 "SUNWARP_EXITS": SUNWARP_EXITS,
471 "SPECIAL_ITEM_IDS": SPECIAL_ITEM_IDS, 501 "SPECIAL_ITEM_IDS": SPECIAL_ITEM_IDS,
472 "PANEL_LOCATION_IDS": PANEL_LOCATION_IDS, 502 "PANEL_LOCATION_IDS": PANEL_LOCATION_IDS,
473 "DOOR_LOCATION_IDS": DOOR_LOCATION_IDS, 503 "DOOR_LOCATION_IDS": DOOR_LOCATION_IDS,