diff options
-rw-r--r-- | player_logic.py | 19 | ||||
-rw-r--r-- | regions.py | 9 | ||||
-rw-r--r-- | rules.py | 9 | ||||
-rw-r--r-- | test/TestMastery.py | 19 |
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 | ||
36 | class PlayerLocation(NamedTuple): | 40 | class 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 | |||
42 | class 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")) | ||