summary refs log tree commit diff stats
diff options
context:
space:
mode:
authorStar Rauchenberger <fefferburbia@gmail.com>2023-11-10 14:07:56 -0500
committerGitHub <noreply@github.com>2023-11-10 13:07:56 -0600
commita43fb727a292bd9476dc8de5685c5b6c38a6a919 (patch)
tree9e702125b47c95fcaf2accf548aba241a9d50282
parentbbbbc71bee25cfd22c5304f98f5a7881383585a3 (diff)
downloadlingo-apworld-a43fb727a292bd9476dc8de5685c5b6c38a6a919.tar.gz
lingo-apworld-a43fb727a292bd9476dc8de5685c5b6c38a6a919.tar.bz2
lingo-apworld-a43fb727a292bd9476dc8de5685c5b6c38a6a919.zip
Lingo: Fix edge case painting shuffle accessibility issues (#2441)
* Lingo: Fix painting shuffle logic issue in The Wise

* Lingo: More generic painting cycle prevention

* Lingo: okay how about now

* Lingo: Consider Owl Hallway blocked painting areas in vanilla doors

* Lingo: so honestly I should've seen this one coming

* Lingo: Refined req_blocked for vanilla doors

* Lingo: Orange Tower Basement is also owl-blocked

* Lingo: Rewrite randomize_paintings to eliminate rerolls

Now, mapping is done in two phases, rather than assigning everything at once and then rerolling if the mapping is non-viable.
-rw-r--r--LL1.yaml11
-rw-r--r--player_logic.py62
-rw-r--r--static_logic.py15
3 files changed, 58 insertions, 30 deletions
diff --git a/LL1.yaml b/LL1.yaml index 7ae015d..db1418f 100644 --- a/LL1.yaml +++ b/LL1.yaml
@@ -97,6 +97,11 @@
97 # Use "required_when_no_doors" instead if it would be 97 # Use "required_when_no_doors" instead if it would be
98 # possible to enter the room without the painting in door 98 # possible to enter the room without the painting in door
99 # shuffle mode. 99 # shuffle mode.
100 # - req_blocked: Marks that a painting cannot be an entrance leading to a
101 # required painting. Paintings within a room that has a
102 # required painting are automatically req blocked.
103 # Use "req_blocked_when_no_doors" instead if it would be
104 # fine in door shuffle mode.
100 # - move: Denotes that the painting is able to move. 105 # - move: Denotes that the painting is able to move.
101 Starting Room: 106 Starting Room:
102 entrances: 107 entrances:
@@ -2210,6 +2215,7 @@
2210 - id: map_painting2 2215 - id: map_painting2
2211 orientation: north 2216 orientation: north
2212 enter_only: True # otherwise you might just skip the whole game! 2217 enter_only: True # otherwise you might just skip the whole game!
2218 req_blocked_when_no_doors: True # owl hallway in vanilla doors
2213 Roof: 2219 Roof:
2214 entrances: 2220 entrances:
2215 Orange Tower Seventh Floor: True 2221 Orange Tower Seventh Floor: True
@@ -2276,6 +2282,7 @@
2276 paintings: 2282 paintings:
2277 - id: arrows_painting_11 2283 - id: arrows_painting_11
2278 orientation: east 2284 orientation: east
2285 req_blocked_when_no_doors: True # owl hallway in vanilla doors
2279 Courtyard: 2286 Courtyard:
2280 entrances: 2287 entrances:
2281 Roof: True 2288 Roof: True
@@ -5755,11 +5762,13 @@
5755 move: True 5762 move: True
5756 required_door: 5763 required_door:
5757 door: Exit 5764 door: Exit
5765 req_blocked_when_no_doors: True # the wondrous (table) in vanilla doors
5758 - id: symmetry_painting_a_6 5766 - id: symmetry_painting_a_6
5759 orientation: west 5767 orientation: west
5760 exit_only: True 5768 exit_only: True
5761 - id: symmetry_painting_b_6 5769 - id: symmetry_painting_b_6
5762 orientation: north 5770 orientation: north
5771 req_blocked_when_no_doors: True # the wondrous (table) in vanilla doors
5763 Arrow Garden: 5772 Arrow Garden:
5764 entrances: 5773 entrances:
5765 The Wondrous: 5774 The Wondrous:
@@ -6914,6 +6923,7 @@
6914 paintings: 6923 paintings:
6915 - id: clock_painting_3 6924 - id: clock_painting_3
6916 orientation: east 6925 orientation: east
6926 req_blocked: True # outside the wise (with or without door shuffle)
6917 The Red: 6927 The Red:
6918 entrances: 6928 entrances:
6919 Roof: True 6929 Roof: True
@@ -7362,6 +7372,7 @@
7362 paintings: 7372 paintings:
7363 - id: hi_solved_painting4 7373 - id: hi_solved_painting4
7364 orientation: south 7374 orientation: south
7375 req_blocked_when_no_doors: True # owl hallway in vanilla doors
7365 Challenge Room: 7376 Challenge Room:
7366 entrances: 7377 entrances:
7367 Welcome Back Area: 7378 Welcome Back Area:
diff --git a/player_logic.py b/player_logic.py index 217ad91..66fe317 100644 --- a/player_logic.py +++ b/player_logic.py
@@ -241,43 +241,46 @@ class LingoPlayerLogic:
241 241
242 door_shuffle = world.options.shuffle_doors 242 door_shuffle = world.options.shuffle_doors
243 243
244 # Determine the set of exit paintings. All required-exit paintings are included, as are all 244 # First, assign mappings to the required-exit paintings. We ensure that req-blocked paintings do not lead to
245 # required-when-no-doors paintings if door shuffle is off. We then fill the set with random other paintings. 245 # required paintings.
246 chosen_exits = [] 246 req_exits = []
247 required_painting_rooms = REQUIRED_PAINTING_ROOMS
247 if door_shuffle == ShuffleDoors.option_none: 248 if door_shuffle == ShuffleDoors.option_none:
248 chosen_exits = [painting_id for painting_id, painting in PAINTINGS.items() 249 required_painting_rooms += REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS
249 if painting.required_when_no_doors] 250 req_exits = [painting_id for painting_id, painting in PAINTINGS.items() if painting.required_when_no_doors]
250 chosen_exits += [painting_id for painting_id, painting in PAINTINGS.items() 251 req_enterable = [painting_id for painting_id, painting in PAINTINGS.items()
251 if painting.exit_only and painting.required] 252 if not painting.exit_only and not painting.disable and not painting.req_blocked and
253 not painting.req_blocked_when_no_doors and painting.room not in required_painting_rooms]
254 else:
255 req_enterable = [painting_id for painting_id, painting in PAINTINGS.items()
256 if not painting.exit_only and not painting.disable and not painting.req_blocked and
257 painting.room not in required_painting_rooms]
258 req_exits += [painting_id for painting_id, painting in PAINTINGS.items()
259 if painting.exit_only and painting.required]
260 req_entrances = world.random.sample(req_enterable, len(req_exits))
261
262 self.PAINTING_MAPPING = dict(zip(req_entrances, req_exits))
263
264 # Next, determine the rest of the exit paintings.
252 exitable = [painting_id for painting_id, painting in PAINTINGS.items() 265 exitable = [painting_id for painting_id, painting in PAINTINGS.items()
253 if not painting.enter_only and not painting.disable and not painting.required] 266 if not painting.enter_only and not painting.disable and painting_id not in req_exits and
254 chosen_exits += world.random.sample(exitable, PAINTING_EXITS - len(chosen_exits)) 267 painting_id not in req_entrances]
268 nonreq_exits = world.random.sample(exitable, PAINTING_EXITS - len(req_exits))
269 chosen_exits = req_exits + nonreq_exits
255 270
256 # Determine the set of entrance paintings. 271 # Determine the rest of the entrance paintings.
257 enterable = [painting_id for painting_id, painting in PAINTINGS.items() 272 enterable = [painting_id for painting_id, painting in PAINTINGS.items()
258 if not painting.exit_only and not painting.disable and painting_id not in chosen_exits] 273 if not painting.exit_only and not painting.disable and painting_id not in chosen_exits and
259 chosen_entrances = world.random.sample(enterable, PAINTING_ENTRANCES) 274 painting_id not in req_entrances]
275 chosen_entrances = world.random.sample(enterable, PAINTING_ENTRANCES - len(req_entrances))
260 276
261 # Create a mapping from entrances to exits. 277 # Assign one entrance to each non-required exit, to ensure that the total number of exits is achieved.
262 for warp_exit in chosen_exits: 278 for warp_exit in nonreq_exits:
263 warp_enter = world.random.choice(chosen_entrances) 279 warp_enter = world.random.choice(chosen_entrances)
264
265 # Check whether this is a warp from a required painting room to another (or the same) required painting
266 # room. This could cause a cycle that would make certain regions inaccessible.
267 warp_exit_room = PAINTINGS[warp_exit].room
268 warp_enter_room = PAINTINGS[warp_enter].room
269
270 required_painting_rooms = REQUIRED_PAINTING_ROOMS
271 if door_shuffle == ShuffleDoors.option_none:
272 required_painting_rooms += REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS
273
274 if warp_exit_room in required_painting_rooms and warp_enter_room in required_painting_rooms:
275 # This shuffling is non-workable. Start over.
276 return False
277
278 chosen_entrances.remove(warp_enter) 280 chosen_entrances.remove(warp_enter)
279 self.PAINTING_MAPPING[warp_enter] = warp_exit 281 self.PAINTING_MAPPING[warp_enter] = warp_exit
280 282
283 # Assign each of the remaining entrances to any required or non-required exit.
281 for warp_enter in chosen_entrances: 284 for warp_enter in chosen_entrances:
282 warp_exit = world.random.choice(chosen_exits) 285 warp_exit = world.random.choice(chosen_exits)
283 self.PAINTING_MAPPING[warp_enter] = warp_exit 286 self.PAINTING_MAPPING[warp_enter] = warp_exit
@@ -292,7 +295,8 @@ class LingoPlayerLogic:
292 # Just for sanity's sake, ensure that all required painting rooms are accessed. 295 # Just for sanity's sake, ensure that all required painting rooms are accessed.
293 for painting_id, painting in PAINTINGS.items(): 296 for painting_id, painting in PAINTINGS.items():
294 if painting_id not in self.PAINTING_MAPPING.values() \ 297 if painting_id not in self.PAINTING_MAPPING.values() \
295 and (painting.required or (painting.required_when_no_doors and door_shuffle == 0)): 298 and (painting.required or (painting.required_when_no_doors and
299 door_shuffle == ShuffleDoors.option_none)):
296 return False 300 return False
297 301
298 return True 302 return True
diff --git a/static_logic.py b/static_logic.py index d122169..f6690f9 100644 --- a/static_logic.py +++ b/static_logic.py
@@ -63,6 +63,8 @@ class Painting(NamedTuple):
63 required_door: Optional[RoomAndDoor] 63 required_door: Optional[RoomAndDoor]
64 disable: bool 64 disable: bool
65 move: bool 65 move: bool
66 req_blocked: bool
67 req_blocked_when_no_doors: bool
66 68
67 69
68class Progression(NamedTuple): 70class Progression(NamedTuple):
@@ -471,6 +473,16 @@ def process_painting(room_name, painting_data):
471 else: 473 else:
472 enter_only = False 474 enter_only = False
473 475
476 if "req_blocked" in painting_data:
477 req_blocked = painting_data["req_blocked"]
478 else:
479 req_blocked = False
480
481 if "req_blocked_when_no_doors" in painting_data:
482 req_blocked_when_no_doors = painting_data["req_blocked_when_no_doors"]
483 else:
484 req_blocked_when_no_doors = False
485
474 required_door = None 486 required_door = None
475 if "required_door" in painting_data: 487 if "required_door" in painting_data:
476 door = painting_data["required_door"] 488 door = painting_data["required_door"]
@@ -480,7 +492,8 @@ def process_painting(room_name, painting_data):
480 ) 492 )
481 493
482 painting_obj = Painting(painting_id, room_name, enter_only, exit_only, orientation, 494 painting_obj = Painting(painting_id, room_name, enter_only, exit_only, orientation,
483 required_painting, rwnd, required_door, disable_painting, move_painting) 495 required_painting, rwnd, required_door, disable_painting, move_painting, req_blocked,
496 req_blocked_when_no_doors)
484 PAINTINGS[painting_id] = painting_obj 497 PAINTINGS[painting_id] = painting_obj
485 PAINTINGS_BY_ROOM[room_name].append(painting_obj) 498 PAINTINGS_BY_ROOM[room_name].append(painting_obj)
486 499