summary refs log tree commit diff stats
diff options
context:
space:
mode:
authorStar Rauchenberger <fefferburbia@gmail.com>2024-06-03 04:51:27 -0400
committerGitHub <noreply@github.com>2024-06-03 03:51:27 -0500
commite75c4b6b26ae2406a40718e084ad15421c9ce3be (patch)
treee94414fc9b3c3d8ca292222b18460b174f2f2e34
parent50e452717500aaaa8ea056f127cc2c4b353c310a (diff)
downloadlingo-apworld-e75c4b6b26ae2406a40718e084ad15421c9ce3be.tar.gz
lingo-apworld-e75c4b6b26ae2406a40718e084ad15421c9ce3be.tar.bz2
lingo-apworld-e75c4b6b26ae2406a40718e084ad15421c9ce3be.zip
Lingo: Fix Basement access with THE MASTER (#3231)
-rw-r--r--player_logic.py19
-rw-r--r--regions.py9
-rw-r--r--rules.py9
-rw-r--r--test/TestMastery.py19
4 files changed, 43 insertions, 13 deletions
diff --git a/player_logic.py b/player_logic.py index b6941f3..1621620 100644 --- a/player_logic.py +++ b/player_logic.py
@@ -18,19 +18,23 @@ class AccessRequirements:
18 rooms: Set[str] 18 rooms: Set[str]
19 doors: Set[RoomAndDoor] 19 doors: Set[RoomAndDoor]
20 colors: Set[str] 20 colors: Set[str]
21 the_master: bool
21 22
22 def __init__(self): 23 def __init__(self):
23 self.rooms = set() 24 self.rooms = set()
24 self.doors = set() 25 self.doors = set()
25 self.colors = set() 26 self.colors = set()
27 self.the_master = False
26 28
27 def merge(self, other: "AccessRequirements"): 29 def merge(self, other: "AccessRequirements"):
28 self.rooms |= other.rooms 30 self.rooms |= other.rooms
29 self.doors |= other.doors 31 self.doors |= other.doors
30 self.colors |= other.colors 32 self.colors |= other.colors
33 self.the_master |= other.the_master
31 34
32 def __str__(self): 35 def __str__(self):
33 return f"AccessRequirements(rooms={self.rooms}, doors={self.doors}, colors={self.colors})" 36 return f"AccessRequirements(rooms={self.rooms}, doors={self.doors}, colors={self.colors})," \
37 f" the_master={self.the_master}"
34 38
35 39
36class PlayerLocation(NamedTuple): 40class PlayerLocation(NamedTuple):
@@ -463,6 +467,9 @@ class LingoPlayerLogic:
463 req_panel.panel, world) 467 req_panel.panel, world)
464 access_reqs.merge(sub_access_reqs) 468 access_reqs.merge(sub_access_reqs)
465 469
470 if panel == "THE MASTER":
471 access_reqs.the_master = True
472
466 self.panel_reqs[room][panel] = access_reqs 473 self.panel_reqs[room][panel] = access_reqs
467 474
468 return self.panel_reqs[room][panel] 475 return self.panel_reqs[room][panel]
@@ -502,15 +509,17 @@ class LingoPlayerLogic:
502 unhindered_panels_by_color: dict[Optional[str], int] = {} 509 unhindered_panels_by_color: dict[Optional[str], int] = {}
503 510
504 for panel_name, panel_data in room_data.items(): 511 for panel_name, panel_data in room_data.items():
505 # We won't count non-counting panels. THE MASTER has special access rules and is handled separately. 512 # We won't count non-counting panels.
506 if panel_data.non_counting or panel_name == "THE MASTER": 513 if panel_data.non_counting:
507 continue 514 continue
508 515
509 # We won't coalesce any panels that have requirements beyond colors. To simplify things for now, we will 516 # We won't coalesce any panels that have requirements beyond colors. To simplify things for now, we will
510 # only coalesce single-color panels. Chains/stacks/combo puzzles will be separate. 517 # only coalesce single-color panels. Chains/stacks/combo puzzles will be separate. THE MASTER has
518 # special access rules and is handled separately.
511 if len(panel_data.required_panels) > 0 or len(panel_data.required_doors) > 0\ 519 if len(panel_data.required_panels) > 0 or len(panel_data.required_doors) > 0\
512 or len(panel_data.required_rooms) > 0\ 520 or len(panel_data.required_rooms) > 0\
513 or (world.options.shuffle_colors and len(panel_data.colors) > 1): 521 or (world.options.shuffle_colors and len(panel_data.colors) > 1)\
522 or panel_name == "THE MASTER":
514 self.counting_panel_reqs.setdefault(room_name, []).append( 523 self.counting_panel_reqs.setdefault(room_name, []).append(
515 (self.calculate_panel_requirements(room_name, panel_name, world), 1)) 524 (self.calculate_panel_requirements(room_name, panel_name, world), 1))
516 else: 525 else:
diff --git a/regions.py b/regions.py index 4b357db..9834f04 100644 --- a/regions.py +++ b/regions.py
@@ -49,8 +49,15 @@ def connect_entrance(regions: Dict[str, Region], source_region: Region, target_r
49 if door is not None: 49 if door is not None:
50 effective_room = target_region.name if door.room is None else door.room 50 effective_room = target_region.name if door.room is None else door.room
51 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, {}):
52 for region in world.player_logic.calculate_door_requirements(effective_room, door.door, world).rooms: 52 access_reqs = world.player_logic.calculate_door_requirements(effective_room, door.door, world)
53 for region in access_reqs.rooms:
53 world.multiworld.register_indirect_condition(regions[region], connection) 54 world.multiworld.register_indirect_condition(regions[region], connection)
55
56 # This pretty much only applies to Orange Tower Sixth Floor -> Orange Tower Basement.
57 if access_reqs.the_master:
58 for mastery_req in world.player_logic.mastery_reqs:
59 for region in mastery_req.rooms:
60 world.multiworld.register_indirect_condition(regions[region], connection)
54 61
55 if not pilgrimage and world.options.enable_pilgrimage and is_acceptable_pilgrimage_entrance(entrance_type, world)\ 62 if not pilgrimage and world.options.enable_pilgrimage and is_acceptable_pilgrimage_entrance(entrance_type, world)\
56 and source_region.name != "Menu": 63 and source_region.name != "Menu":
diff --git a/rules.py b/rules.py index 9cc11fd..d91c53f 100644 --- a/rules.py +++ b/rules.py
@@ -42,12 +42,6 @@ def lingo_can_use_level_2_location(state: CollectionState, world: "LingoWorld"):
42 counted_panels += panel_count 42 counted_panels += panel_count
43 if counted_panels >= world.options.level_2_requirement.value - 1: 43 if counted_panels >= world.options.level_2_requirement.value - 1:
44 return True 44 return True
45 # THE MASTER has to be handled separately, because it has special access rules.
46 if state.can_reach("Orange Tower Seventh Floor", "Region", world.player)\
47 and lingo_can_use_mastery_location(state, world):
48 counted_panels += 1
49 if counted_panels >= world.options.level_2_requirement.value - 1:
50 return True
51 return False 45 return False
52 46
53 47
@@ -65,6 +59,9 @@ def _lingo_can_satisfy_requirements(state: CollectionState, access: AccessRequir
65 if not state.has(color.capitalize(), world.player): 59 if not state.has(color.capitalize(), world.player):
66 return False 60 return False
67 61
62 if access.the_master and not lingo_can_use_mastery_location(state, world):
63 return False
64
68 return True 65 return True
69 66
70 67
diff --git a/test/TestMastery.py b/test/TestMastery.py index 3fb3c95..6e56339 100644 --- a/test/TestMastery.py +++ b/test/TestMastery.py
@@ -36,4 +36,21 @@ class TestMasteryWhenVictoryIsTheMaster(LingoTestBase):
36 self.assertFalse(self.can_reach_location("Orange Tower Seventh Floor - Mastery Achievements")) 36 self.assertFalse(self.can_reach_location("Orange Tower Seventh Floor - Mastery Achievements"))
37 37
38 self.collect_by_name(["Green", "Gray", "Brown", "Yellow"]) 38 self.collect_by_name(["Green", "Gray", "Brown", "Yellow"])
39 self.assertTrue(self.can_reach_location("Orange Tower Seventh Floor - Mastery Achievements")) \ No newline at end of file 39 self.assertTrue(self.can_reach_location("Orange Tower Seventh Floor - Mastery Achievements"))
40
41
42class TestMasteryBlocksDependents(LingoTestBase):
43 options = {
44 "mastery_achievements": "24",
45 "shuffle_colors": "true",
46 "location_checks": "insanity"
47 }
48
49 def test_requirement(self):
50 self.collect_all_but("Gray")
51 self.assertFalse(self.can_reach_location("Orange Tower Basement - THE LIBRARY"))
52 self.assertFalse(self.can_reach_location("Orange Tower Seventh Floor - MASTERY"))
53
54 self.collect_by_name("Gray")
55 self.assertTrue(self.can_reach_location("Orange Tower Basement - THE LIBRARY"))
56 self.assertTrue(self.can_reach_location("Orange Tower Seventh Floor - MASTERY"))