diff options
author | Star Rauchenberger <fefferburbia@gmail.com> | 2024-04-18 11:45:33 -0500 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-04-18 18:45:33 +0200 |
commit | b45b9bd74612239ebc29322fc340b1bee62e7032 (patch) | |
tree | d61b0077586fa146a418a46536daef02ae27989a | |
parent | 7a358cedc44c0892a4c369a4884a23a001535d63 (diff) | |
download | lingo-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.
-rw-r--r-- | __init__.py | 6 | ||||
-rw-r--r-- | data/LL1.yaml | 587 | ||||
-rw-r--r-- | data/generated.dat | bin | 130691 -> 135088 bytes | |||
-rw-r--r-- | data/ids.yaml | 35 | ||||
-rw-r--r-- | datatypes.py | 19 | ||||
-rw-r--r-- | items.py | 39 | ||||
-rw-r--r-- | locations.py | 2 | ||||
-rw-r--r-- | options.py | 47 | ||||
-rw-r--r-- | player_logic.py | 106 | ||||
-rw-r--r-- | regions.py | 105 | ||||
-rw-r--r-- | rules.py | 4 | ||||
-rw-r--r-- | static_logic.py | 8 | ||||
-rw-r--r-- | test/TestOptions.py | 20 | ||||
-rw-r--r-- | test/TestPilgrimage.py | 114 | ||||
-rw-r--r-- | test/TestSunwarps.py | 213 | ||||
-rw-r--r-- | utils/pickle_static_data.py | 96 | ||||
-rw-r--r-- | utils/validate_config.rb | 48 |
17 files changed, 1152 insertions, 297 deletions
diff --git a/__init__.py b/__init__.py index 25be166..537b149 100644 --- a/__init__.py +++ b/__init__.py | |||
@@ -132,7 +132,8 @@ class LingoWorld(World): | |||
132 | def fill_slot_data(self): | 132 | def fill_slot_data(self): |
133 | slot_options = [ | 133 | slot_options = [ |
134 | "death_link", "victory_condition", "shuffle_colors", "shuffle_doors", "shuffle_paintings", "shuffle_panels", | 134 | "death_link", "victory_condition", "shuffle_colors", "shuffle_doors", "shuffle_paintings", "shuffle_panels", |
135 | "mastery_achievements", "level_2_requirement", "location_checks", "early_color_hallways" | 135 | "enable_pilgrimage", "sunwarp_access", "mastery_achievements", "level_2_requirement", "location_checks", |
136 | "early_color_hallways", "pilgrimage_allows_roof_access", "pilgrimage_allows_paintings", "shuffle_sunwarps" | ||
136 | ] | 137 | ] |
137 | 138 | ||
138 | slot_data = { | 139 | slot_data = { |
@@ -143,6 +144,9 @@ class LingoWorld(World): | |||
143 | if self.options.shuffle_paintings: | 144 | if self.options.shuffle_paintings: |
144 | slot_data["painting_entrance_to_exit"] = self.player_logic.painting_mapping | 145 | slot_data["painting_entrance_to_exit"] = self.player_logic.painting_mapping |
145 | 146 | ||
147 | if self.options.shuffle_sunwarps: | ||
148 | slot_data["sunwarp_permutation"] = self.player_logic.sunwarp_mapping | ||
149 | |||
146 | return slot_data | 150 | return slot_data |
147 | 151 | ||
148 | def get_filler_item_name(self) -> str: | 152 | def get_filler_item_name(self) -> str: |
diff --git a/data/LL1.yaml b/data/LL1.yaml index f2d2a9f..c33cad3 100644 --- a/data/LL1.yaml +++ b/data/LL1.yaml | |||
@@ -54,6 +54,8 @@ | |||
54 | # this door will open the doors listed here. | 54 | # this door will open the doors listed here. |
55 | # - painting_id: An internal ID of a painting that should be moved upon | 55 | # - painting_id: An internal ID of a painting that should be moved upon |
56 | # receiving this door. | 56 | # receiving this door. |
57 | # - warp_id: An internal ID or IDs of warps that should be disabled | ||
58 | # until receiving this door. | ||
57 | # - panels: These are the panels that canonically open this door. If | 59 | # - panels: These are the panels that canonically open this door. If |
58 | # there is only one panel for the door, then that panel is a | 60 | # there is only one panel for the door, then that panel is a |
59 | # check. If there is more than one panel, then that entire | 61 | # check. If there is more than one panel, then that entire |
@@ -73,10 +75,6 @@ | |||
73 | # will be covered by a single item. | 75 | # will be covered by a single item. |
74 | # - include_reduce: Door checks are assumed to be EXCLUDED when reduce checks | 76 | # - include_reduce: Door checks are assumed to be EXCLUDED when reduce checks |
75 | # is on. This option includes the check anyway. | 77 | # is on. This option includes the check anyway. |
76 | # - junk_item: If on, the item for this door will be considered a junk | ||
77 | # item instead of a progression item. Only use this for | ||
78 | # doors that could never gate progression regardless of | ||
79 | # options and state. | ||
80 | # - event: Denotes that the door is event only. This is similar to | 78 | # - event: Denotes that the door is event only. This is similar to |
81 | # setting both skip_location and skip_item. | 79 | # setting both skip_location and skip_item. |
82 | # | 80 | # |
@@ -106,9 +104,42 @@ | |||
106 | # Use "req_blocked_when_no_doors" instead if it would be | 104 | # Use "req_blocked_when_no_doors" instead if it would be |
107 | # fine in door shuffle mode. | 105 | # fine in door shuffle mode. |
108 | # - move: Denotes that the painting is able to move. | 106 | # - move: Denotes that the painting is able to move. |
107 | # | ||
108 | # sunwarps is an array of sunwarps in the room. This is used for sunwarp | ||
109 | # shuffling. | ||
110 | # - dots: The number of dots on this sunwarp. | ||
111 | # - direction: "enter" or "exit" | ||
112 | # - entrance_indicator_pos: Coordinates for where the entrance indicator | ||
113 | # should be placed if this becomes an entrance. | ||
114 | # - orientation: One of north/south/east/west. | ||
109 | Starting Room: | 115 | Starting Room: |
110 | entrances: | 116 | entrances: |
111 | Menu: True | 117 | Menu: |
118 | warp: True | ||
119 | Outside The Wise: | ||
120 | painting: True | ||
121 | Rhyme Room (Circle): | ||
122 | painting: True | ||
123 | Rhyme Room (Target): | ||
124 | painting: True | ||
125 | Wondrous Lobby: | ||
126 | painting: True | ||
127 | Orange Tower Third Floor: | ||
128 | painting: True | ||
129 | Color Hunt: | ||
130 | painting: True | ||
131 | Owl Hallway: | ||
132 | painting: True | ||
133 | The Wondrous: | ||
134 | room: The Wondrous | ||
135 | door: Exit | ||
136 | painting: True | ||
137 | Orange Tower Sixth Floor: | ||
138 | painting: True | ||
139 | Orange Tower Basement: | ||
140 | painting: True | ||
141 | The Colorful: | ||
142 | painting: True | ||
112 | panels: | 143 | panels: |
113 | HI: | 144 | HI: |
114 | id: Entry Room/Panel_hi_hi | 145 | id: Entry Room/Panel_hi_hi |
@@ -416,7 +447,7 @@ | |||
416 | The Traveled: | 447 | The Traveled: |
417 | door: Traveled Entrance | 448 | door: Traveled Entrance |
418 | Roof: True # through the sunwarp | 449 | Roof: True # through the sunwarp |
419 | Outside The Undeterred: # (NOTE: used in hardcoded pilgrimage) | 450 | Outside The Undeterred: |
420 | room: Outside The Undeterred | 451 | room: Outside The Undeterred |
421 | door: Green Painting | 452 | door: Green Painting |
422 | painting: True | 453 | painting: True |
@@ -500,6 +531,11 @@ | |||
500 | paintings: | 531 | paintings: |
501 | - id: maze_painting | 532 | - id: maze_painting |
502 | orientation: west | 533 | orientation: west |
534 | sunwarps: | ||
535 | - dots: 1 | ||
536 | direction: enter | ||
537 | entrance_indicator_pos: [18, 2.5, -17.01] | ||
538 | orientation: north | ||
503 | Dead End Area: | 539 | Dead End Area: |
504 | entrances: | 540 | entrances: |
505 | Hidden Room: | 541 | Hidden Room: |
@@ -526,12 +562,52 @@ | |||
526 | paintings: | 562 | paintings: |
527 | - id: smile_painting_6 | 563 | - id: smile_painting_6 |
528 | orientation: north | 564 | orientation: north |
529 | Pilgrim Antechamber: | 565 | Sunwarps: |
530 | # Let's not shuffle the paintings yet. | 566 | # This is a special, meta-ish room. |
531 | entrances: | 567 | entrances: |
532 | # The pilgrimage is hardcoded in rules.py | 568 | Menu: True |
533 | Starting Room: | 569 | doors: |
534 | door: Sun Painting | 570 | 1 Sunwarp: |
571 | warp_id: Teleporter Warps/Sunwarp_enter_1 | ||
572 | door_group: Sunwarps | ||
573 | skip_location: True | ||
574 | item_name: "1 Sunwarp" | ||
575 | 2 Sunwarp: | ||
576 | warp_id: Teleporter Warps/Sunwarp_enter_2 | ||
577 | door_group: Sunwarps | ||
578 | skip_location: True | ||
579 | item_name: 2 Sunwarp | ||
580 | 3 Sunwarp: | ||
581 | warp_id: Teleporter Warps/Sunwarp_enter_3 | ||
582 | door_group: Sunwarps | ||
583 | skip_location: True | ||
584 | item_name: "3 Sunwarp" | ||
585 | 4 Sunwarp: | ||
586 | warp_id: Teleporter Warps/Sunwarp_enter_4 | ||
587 | door_group: Sunwarps | ||
588 | skip_location: True | ||
589 | item_name: 4 Sunwarp | ||
590 | 5 Sunwarp: | ||
591 | warp_id: Teleporter Warps/Sunwarp_enter_5 | ||
592 | door_group: Sunwarps | ||
593 | skip_location: True | ||
594 | item_name: "5 Sunwarp" | ||
595 | 6 Sunwarp: | ||
596 | warp_id: Teleporter Warps/Sunwarp_enter_6 | ||
597 | door_group: Sunwarps | ||
598 | skip_location: True | ||
599 | item_name: "6 Sunwarp" | ||
600 | progression: | ||
601 | Progressive Pilgrimage: | ||
602 | - 1 Sunwarp | ||
603 | - 2 Sunwarp | ||
604 | - 3 Sunwarp | ||
605 | - 4 Sunwarp | ||
606 | - 5 Sunwarp | ||
607 | - 6 Sunwarp | ||
608 | Pilgrim Antechamber: | ||
609 | # The entrances to this room are special. When pilgrimage is enabled, we use a special access rule to determine | ||
610 | # whether a pilgrimage can succeed. When pilgrimage is disabled, the sun painting will be added to the pool. | ||
535 | panels: | 611 | panels: |
536 | HOT CRUST: | 612 | HOT CRUST: |
537 | id: Lingo Room/Panel_shortcut | 613 | id: Lingo Room/Panel_shortcut |
@@ -541,6 +617,7 @@ | |||
541 | id: Lingo Room/Panel_pilgrim | 617 | id: Lingo Room/Panel_pilgrim |
542 | colors: blue | 618 | colors: blue |
543 | tag: midblue | 619 | tag: midblue |
620 | check: True | ||
544 | MASTERY: | 621 | MASTERY: |
545 | id: Master Room/Panel_mastery_mastery14 | 622 | id: Master Room/Panel_mastery_mastery14 |
546 | tag: midwhite | 623 | tag: midwhite |
@@ -636,11 +713,19 @@ | |||
636 | - THIS | 713 | - THIS |
637 | Crossroads: | 714 | Crossroads: |
638 | entrances: | 715 | entrances: |
639 | Hub Room: True # The sunwarp means that we never need the ORDER door | 716 | Hub Room: |
640 | Color Hallways: True | 717 | - room: Sunwarps |
718 | door: 1 Sunwarp | ||
719 | sunwarp: True | ||
720 | - room: Hub Room | ||
721 | door: Crossroads Entrance | ||
722 | Color Hallways: | ||
723 | warp: True | ||
641 | The Tenacious: | 724 | The Tenacious: |
642 | door: Tenacious Entrance | 725 | door: Tenacious Entrance |
643 | Orange Tower Fourth Floor: True # through IRK HORN | 726 | Orange Tower Fourth Floor: |
727 | - warp: True # through IRK HORN | ||
728 | - door: Tower Entrance | ||
644 | Amen Name Area: | 729 | Amen Name Area: |
645 | room: Lost Area | 730 | room: Lost Area |
646 | door: Exit | 731 | door: Exit |
@@ -760,7 +845,6 @@ | |||
760 | - SWORD | 845 | - SWORD |
761 | Eye Wall: | 846 | Eye Wall: |
762 | id: Shuffle Room Area Doors/Door_behind | 847 | id: Shuffle Room Area Doors/Door_behind |
763 | junk_item: True | ||
764 | door_group: Crossroads Doors | 848 | door_group: Crossroads Doors |
765 | panels: | 849 | panels: |
766 | - BEND HI | 850 | - BEND HI |
@@ -795,6 +879,11 @@ | |||
795 | door: Eye Wall | 879 | door: Eye Wall |
796 | - id: smile_painting_4 | 880 | - id: smile_painting_4 |
797 | orientation: south | 881 | orientation: south |
882 | sunwarps: | ||
883 | - dots: 1 | ||
884 | direction: exit | ||
885 | entrance_indicator_pos: [ -17, 2.5, -41.01 ] | ||
886 | orientation: north | ||
798 | Lost Area: | 887 | Lost Area: |
799 | entrances: | 888 | entrances: |
800 | Outside The Agreeable: | 889 | Outside The Agreeable: |
@@ -1036,11 +1125,12 @@ | |||
1036 | - LEAF | 1125 | - LEAF |
1037 | - FEEL | 1126 | - FEEL |
1038 | Outside The Agreeable: | 1127 | Outside The Agreeable: |
1039 | # Let's ignore the blue warp thing for now because the lookout is a dead | ||
1040 | # end. Later on it could be filler checks. | ||
1041 | entrances: | 1128 | entrances: |
1042 | # We don't have to list Lost Area because of Crossroads. | 1129 | Crossroads: |
1043 | Crossroads: True | 1130 | warp: True |
1131 | Lost Area: | ||
1132 | room: Lost Area | ||
1133 | door: Exit | ||
1044 | The Tenacious: | 1134 | The Tenacious: |
1045 | door: Tenacious Entrance | 1135 | door: Tenacious Entrance |
1046 | The Agreeable: | 1136 | The Agreeable: |
@@ -1053,12 +1143,11 @@ | |||
1053 | Starting Room: | 1143 | Starting Room: |
1054 | door: Painting Shortcut | 1144 | door: Painting Shortcut |
1055 | painting: True | 1145 | painting: True |
1056 | Hallway Room (2): True | 1146 | Hallway Room (1): |
1057 | Hallway Room (3): True | 1147 | warp: True |
1058 | Hallway Room (4): True | ||
1059 | Hedge Maze: True # through the door to the sectioned-off part of the hedge maze | 1148 | Hedge Maze: True # through the door to the sectioned-off part of the hedge maze |
1060 | Cellar: | 1149 | Compass Room: |
1061 | door: Lookout Entrance | 1150 | warp: True |
1062 | panels: | 1151 | panels: |
1063 | MASSACRED: | 1152 | MASSACRED: |
1064 | id: Palindrome Room/Panel_massacred_sacred | 1153 | id: Palindrome Room/Panel_massacred_sacred |
@@ -1104,11 +1193,6 @@ | |||
1104 | required_door: | 1193 | required_door: |
1105 | room: Outside The Undeterred | 1194 | room: Outside The Undeterred |
1106 | door: Fives | 1195 | door: Fives |
1107 | OUT: | ||
1108 | id: Hallway Room/Panel_out_out | ||
1109 | check: True | ||
1110 | exclude_reduce: True | ||
1111 | tag: midwhite | ||
1112 | HIDE: | 1196 | HIDE: |
1113 | id: Maze Room/Panel_hide_seek_4 | 1197 | id: Maze Room/Panel_hide_seek_4 |
1114 | colors: black | 1198 | colors: black |
@@ -1117,52 +1201,6 @@ | |||
1117 | id: Maze Room/Panel_daze_maze | 1201 | id: Maze Room/Panel_daze_maze |
1118 | colors: purple | 1202 | colors: purple |
1119 | tag: midpurp | 1203 | tag: midpurp |
1120 | WALL: | ||
1121 | id: Hallway Room/Panel_castle_1 | ||
1122 | colors: blue | ||
1123 | tag: quad bot blue | ||
1124 | link: qbb CASTLE | ||
1125 | KEEP: | ||
1126 | id: Hallway Room/Panel_castle_2 | ||
1127 | colors: blue | ||
1128 | tag: quad bot blue | ||
1129 | link: qbb CASTLE | ||
1130 | BAILEY: | ||
1131 | id: Hallway Room/Panel_castle_3 | ||
1132 | colors: blue | ||
1133 | tag: quad bot blue | ||
1134 | link: qbb CASTLE | ||
1135 | TOWER: | ||
1136 | id: Hallway Room/Panel_castle_4 | ||
1137 | colors: blue | ||
1138 | tag: quad bot blue | ||
1139 | link: qbb CASTLE | ||
1140 | NORTH: | ||
1141 | id: Cross Room/Panel_north_missing | ||
1142 | colors: green | ||
1143 | tag: forbid | ||
1144 | required_panel: | ||
1145 | - room: Outside The Bold | ||
1146 | panel: SOUND | ||
1147 | - room: Outside The Bold | ||
1148 | panel: YEAST | ||
1149 | - room: Outside The Bold | ||
1150 | panel: WET | ||
1151 | DIAMONDS: | ||
1152 | id: Cross Room/Panel_diamonds_missing | ||
1153 | colors: green | ||
1154 | tag: forbid | ||
1155 | required_room: Suits Area | ||
1156 | FIRE: | ||
1157 | id: Cross Room/Panel_fire_missing | ||
1158 | colors: green | ||
1159 | tag: forbid | ||
1160 | required_room: Elements Area | ||
1161 | WINTER: | ||
1162 | id: Cross Room/Panel_winter_missing | ||
1163 | colors: green | ||
1164 | tag: forbid | ||
1165 | required_room: Orange Tower Fifth Floor | ||
1166 | doors: | 1204 | doors: |
1167 | Tenacious Entrance: | 1205 | Tenacious Entrance: |
1168 | id: Palindrome Room Area Doors/Door_massacred_sacred | 1206 | id: Palindrome Room Area Doors/Door_massacred_sacred |
@@ -1194,15 +1232,49 @@ | |||
1194 | panels: | 1232 | panels: |
1195 | - room: Color Hunt | 1233 | - room: Color Hunt |
1196 | panel: PURPLE | 1234 | panel: PURPLE |
1197 | Hallway Door: | 1235 | paintings: |
1198 | id: Red Blue Purple Room Area Doors/Door_room_2 | 1236 | - id: eyes_yellow_painting |
1199 | door_group: Hallway Room Doors | 1237 | orientation: east |
1200 | location_name: Hallway Room - First Room | 1238 | sunwarps: |
1201 | panels: | 1239 | - dots: 6 |
1202 | - WALL | 1240 | direction: enter |
1203 | - KEEP | 1241 | entrance_indicator_pos: [ 3, 2.5, -55.01 ] |
1204 | - BAILEY | 1242 | orientation: north |
1205 | - TOWER | 1243 | Compass Room: |
1244 | entrances: | ||
1245 | Outside The Agreeable: | ||
1246 | warp: True | ||
1247 | Cellar: | ||
1248 | door: Lookout Entrance | ||
1249 | warp: True | ||
1250 | panels: | ||
1251 | NORTH: | ||
1252 | id: Cross Room/Panel_north_missing | ||
1253 | colors: green | ||
1254 | tag: forbid | ||
1255 | required_panel: | ||
1256 | - room: Outside The Bold | ||
1257 | panel: SOUND | ||
1258 | - room: Outside The Bold | ||
1259 | panel: YEAST | ||
1260 | - room: Outside The Bold | ||
1261 | panel: WET | ||
1262 | DIAMONDS: | ||
1263 | id: Cross Room/Panel_diamonds_missing | ||
1264 | colors: green | ||
1265 | tag: forbid | ||
1266 | required_room: Suits Area | ||
1267 | FIRE: | ||
1268 | id: Cross Room/Panel_fire_missing | ||
1269 | colors: green | ||
1270 | tag: forbid | ||
1271 | required_room: Elements Area | ||
1272 | WINTER: | ||
1273 | id: Cross Room/Panel_winter_missing | ||
1274 | colors: green | ||
1275 | tag: forbid | ||
1276 | required_room: Orange Tower Fifth Floor | ||
1277 | doors: | ||
1206 | Lookout Entrance: | 1278 | Lookout Entrance: |
1207 | id: Cross Room Doors/Door_missing | 1279 | id: Cross Room Doors/Door_missing |
1208 | location_name: Outside The Agreeable - Lookout Panels | 1280 | location_name: Outside The Agreeable - Lookout Panels |
@@ -1212,21 +1284,8 @@ | |||
1212 | - DIAMONDS | 1284 | - DIAMONDS |
1213 | - FIRE | 1285 | - FIRE |
1214 | paintings: | 1286 | paintings: |
1215 | - id: panda_painting | ||
1216 | orientation: south | ||
1217 | - id: eyes_yellow_painting | ||
1218 | orientation: east | ||
1219 | - id: pencil_painting7 | 1287 | - id: pencil_painting7 |
1220 | orientation: north | 1288 | orientation: north |
1221 | progression: | ||
1222 | Progressive Hallway Room: | ||
1223 | - Hallway Door | ||
1224 | - room: Hallway Room (2) | ||
1225 | door: Exit | ||
1226 | - room: Hallway Room (3) | ||
1227 | door: Exit | ||
1228 | - room: Hallway Room (4) | ||
1229 | door: Exit | ||
1230 | Dread Hallway: | 1289 | Dread Hallway: |
1231 | entrances: | 1290 | entrances: |
1232 | Outside The Agreeable: | 1291 | Outside The Agreeable: |
@@ -1321,7 +1380,8 @@ | |||
1321 | Hub Room: | 1380 | Hub Room: |
1322 | room: Hub Room | 1381 | room: Hub Room |
1323 | door: Shortcut to Hedge Maze | 1382 | door: Shortcut to Hedge Maze |
1324 | Color Hallways: True | 1383 | Color Hallways: |
1384 | warp: True | ||
1325 | The Agreeable: | 1385 | The Agreeable: |
1326 | room: The Agreeable | 1386 | room: The Agreeable |
1327 | door: Shortcut to Hedge Maze | 1387 | door: Shortcut to Hedge Maze |
@@ -1465,7 +1525,8 @@ | |||
1465 | orientation: north | 1525 | orientation: north |
1466 | The Fearless (First Floor): | 1526 | The Fearless (First Floor): |
1467 | entrances: | 1527 | entrances: |
1468 | The Perceptive: True | 1528 | The Perceptive: |
1529 | warp: True | ||
1469 | panels: | 1530 | panels: |
1470 | SPAN: | 1531 | SPAN: |
1471 | id: Naps Room/Panel_naps_span | 1532 | id: Naps Room/Panel_naps_span |
@@ -1508,6 +1569,7 @@ | |||
1508 | The Fearless (First Floor): | 1569 | The Fearless (First Floor): |
1509 | room: The Fearless (First Floor) | 1570 | room: The Fearless (First Floor) |
1510 | door: Second Floor | 1571 | door: Second Floor |
1572 | warp: True | ||
1511 | panels: | 1573 | panels: |
1512 | NONE: | 1574 | NONE: |
1513 | id: Naps Room/Panel_one_many | 1575 | id: Naps Room/Panel_one_many |
@@ -1557,6 +1619,7 @@ | |||
1557 | The Fearless (First Floor): | 1619 | The Fearless (First Floor): |
1558 | room: The Fearless (Second Floor) | 1620 | room: The Fearless (Second Floor) |
1559 | door: Third Floor | 1621 | door: Third Floor |
1622 | warp: True | ||
1560 | panels: | 1623 | panels: |
1561 | Achievement: | 1624 | Achievement: |
1562 | id: Countdown Panels/Panel_fearless_fearless | 1625 | id: Countdown Panels/Panel_fearless_fearless |
@@ -1585,7 +1648,8 @@ | |||
1585 | Hedge Maze: | 1648 | Hedge Maze: |
1586 | room: Hedge Maze | 1649 | room: Hedge Maze |
1587 | door: Observant Entrance | 1650 | door: Observant Entrance |
1588 | The Incomparable: True | 1651 | The Incomparable: |
1652 | warp: True | ||
1589 | panels: | 1653 | panels: |
1590 | Achievement: | 1654 | Achievement: |
1591 | id: Countdown Panels/Panel_observant_observant | 1655 | id: Countdown Panels/Panel_observant_observant |
@@ -1709,7 +1773,8 @@ | |||
1709 | - SIX | 1773 | - SIX |
1710 | The Incomparable: | 1774 | The Incomparable: |
1711 | entrances: | 1775 | entrances: |
1712 | The Observant: True # Assuming that access to The Observant includes access to the right entrance | 1776 | The Observant: |
1777 | warp: True | ||
1713 | Eight Room: True | 1778 | Eight Room: True |
1714 | Eight Alcove: | 1779 | Eight Alcove: |
1715 | door: Eight Door | 1780 | door: Eight Door |
@@ -1911,9 +1976,11 @@ | |||
1911 | Outside The Wanderer: | 1976 | Outside The Wanderer: |
1912 | room: Outside The Wanderer | 1977 | room: Outside The Wanderer |
1913 | door: Tower Entrance | 1978 | door: Tower Entrance |
1979 | warp: True | ||
1914 | Orange Tower Second Floor: | 1980 | Orange Tower Second Floor: |
1915 | room: Orange Tower | 1981 | room: Orange Tower |
1916 | door: Second Floor | 1982 | door: Second Floor |
1983 | warp: True | ||
1917 | Directional Gallery: | 1984 | Directional Gallery: |
1918 | door: Salt Pepper Door | 1985 | door: Salt Pepper Door |
1919 | Roof: True # through the sunwarp | 1986 | Roof: True # through the sunwarp |
@@ -1944,15 +2011,23 @@ | |||
1944 | - SALT | 2011 | - SALT |
1945 | - room: Directional Gallery | 2012 | - room: Directional Gallery |
1946 | panel: PEPPER | 2013 | panel: PEPPER |
2014 | sunwarps: | ||
2015 | - dots: 4 | ||
2016 | direction: enter | ||
2017 | entrance_indicator_pos: [ -32, 2.5, -14.99 ] | ||
2018 | orientation: south | ||
1947 | Orange Tower Second Floor: | 2019 | Orange Tower Second Floor: |
1948 | entrances: | 2020 | entrances: |
1949 | Orange Tower First Floor: | 2021 | Orange Tower First Floor: |
1950 | room: Orange Tower | 2022 | room: Orange Tower |
1951 | door: Second Floor | 2023 | door: Second Floor |
2024 | warp: True | ||
1952 | Orange Tower Third Floor: | 2025 | Orange Tower Third Floor: |
1953 | room: Orange Tower | 2026 | room: Orange Tower |
1954 | door: Third Floor | 2027 | door: Third Floor |
1955 | Outside The Undeterred: True | 2028 | warp: True |
2029 | Outside The Undeterred: | ||
2030 | warp: True | ||
1956 | Orange Tower Third Floor: | 2031 | Orange Tower Third Floor: |
1957 | entrances: | 2032 | entrances: |
1958 | Knight Night Exit: | 2033 | Knight Night Exit: |
@@ -1961,16 +2036,22 @@ | |||
1961 | Orange Tower Second Floor: | 2036 | Orange Tower Second Floor: |
1962 | room: Orange Tower | 2037 | room: Orange Tower |
1963 | door: Third Floor | 2038 | door: Third Floor |
2039 | warp: True | ||
1964 | Orange Tower Fourth Floor: | 2040 | Orange Tower Fourth Floor: |
1965 | room: Orange Tower | 2041 | room: Orange Tower |
1966 | door: Fourth Floor | 2042 | door: Fourth Floor |
1967 | Hot Crusts Area: True # sunwarp | 2043 | warp: True |
1968 | Bearer Side Area: # This is complicated because of The Bearer's topology | 2044 | Hot Crusts Area: |
2045 | room: Sunwarps | ||
2046 | door: 2 Sunwarp | ||
2047 | sunwarp: True | ||
2048 | Bearer Side Area: | ||
1969 | room: Bearer Side Area | 2049 | room: Bearer Side Area |
1970 | door: Shortcut to Tower | 2050 | door: Shortcut to Tower |
1971 | Rhyme Room (Smiley): | 2051 | Rhyme Room (Smiley): |
1972 | door: Rhyme Room Entrance | 2052 | door: Rhyme Room Entrance |
1973 | Art Gallery: True # mark this as a warp in the sunwarps branch | 2053 | Art Gallery: |
2054 | warp: True | ||
1974 | panels: | 2055 | panels: |
1975 | RED: | 2056 | RED: |
1976 | id: Color Arrow Room/Panel_red_afar | 2057 | id: Color Arrow Room/Panel_red_afar |
@@ -2019,14 +2100,25 @@ | |||
2019 | orientation: east | 2100 | orientation: east |
2020 | - id: flower_painting_5 | 2101 | - id: flower_painting_5 |
2021 | orientation: south | 2102 | orientation: south |
2103 | sunwarps: | ||
2104 | - dots: 2 | ||
2105 | direction: exit | ||
2106 | entrance_indicator_pos: [ 24.01, 2.5, 38 ] | ||
2107 | orientation: west | ||
2108 | - dots: 3 | ||
2109 | direction: enter | ||
2110 | entrance_indicator_pos: [ 28.01, 2.5, 29 ] | ||
2111 | orientation: west | ||
2022 | Orange Tower Fourth Floor: | 2112 | Orange Tower Fourth Floor: |
2023 | entrances: | 2113 | entrances: |
2024 | Orange Tower Third Floor: | 2114 | Orange Tower Third Floor: |
2025 | room: Orange Tower | 2115 | room: Orange Tower |
2026 | door: Fourth Floor | 2116 | door: Fourth Floor |
2117 | warp: True | ||
2027 | Orange Tower Fifth Floor: | 2118 | Orange Tower Fifth Floor: |
2028 | room: Orange Tower | 2119 | room: Orange Tower |
2029 | door: Fifth Floor | 2120 | door: Fifth Floor |
2121 | warp: True | ||
2030 | Hot Crusts Area: | 2122 | Hot Crusts Area: |
2031 | door: Hot Crusts Door | 2123 | door: Hot Crusts Door |
2032 | Crossroads: | 2124 | Crossroads: |
@@ -2034,7 +2126,10 @@ | |||
2034 | door: Tower Entrance | 2126 | door: Tower Entrance |
2035 | - room: Crossroads | 2127 | - room: Crossroads |
2036 | door: Tower Back Entrance | 2128 | door: Tower Back Entrance |
2037 | Courtyard: True | 2129 | Courtyard: |
2130 | - warp: True | ||
2131 | - room: Crossroads | ||
2132 | door: Tower Entrance | ||
2038 | Roof: True # through the sunwarp | 2133 | Roof: True # through the sunwarp |
2039 | panels: | 2134 | panels: |
2040 | RUNT (1): | 2135 | RUNT (1): |
@@ -2067,6 +2162,11 @@ | |||
2067 | id: Shuffle Room Area Doors/Door_hotcrust_shortcuts | 2162 | id: Shuffle Room Area Doors/Door_hotcrust_shortcuts |
2068 | panels: | 2163 | panels: |
2069 | - HOT CRUSTS | 2164 | - HOT CRUSTS |
2165 | sunwarps: | ||
2166 | - dots: 5 | ||
2167 | direction: enter | ||
2168 | entrance_indicator_pos: [ -20, 3, -64.01 ] | ||
2169 | orientation: north | ||
2070 | Hot Crusts Area: | 2170 | Hot Crusts Area: |
2071 | entrances: | 2171 | entrances: |
2072 | Orange Tower Fourth Floor: | 2172 | Orange Tower Fourth Floor: |
@@ -2084,28 +2184,31 @@ | |||
2084 | paintings: | 2184 | paintings: |
2085 | - id: smile_painting_8 | 2185 | - id: smile_painting_8 |
2086 | orientation: north | 2186 | orientation: north |
2187 | sunwarps: | ||
2188 | - dots: 2 | ||
2189 | direction: enter | ||
2190 | entrance_indicator_pos: [ -26, 3.5, -80.01 ] | ||
2191 | orientation: north | ||
2087 | Orange Tower Fifth Floor: | 2192 | Orange Tower Fifth Floor: |
2088 | entrances: | 2193 | entrances: |
2089 | Orange Tower Fourth Floor: | 2194 | Orange Tower Fourth Floor: |
2090 | room: Orange Tower | 2195 | room: Orange Tower |
2091 | door: Fifth Floor | 2196 | door: Fifth Floor |
2197 | warp: True | ||
2092 | Orange Tower Sixth Floor: | 2198 | Orange Tower Sixth Floor: |
2093 | room: Orange Tower | 2199 | room: Orange Tower |
2094 | door: Sixth Floor | 2200 | door: Sixth Floor |
2201 | warp: True | ||
2095 | Cellar: | 2202 | Cellar: |
2096 | room: Room Room | 2203 | room: Room Room |
2097 | door: Cellar Exit | 2204 | door: Cellar Exit |
2205 | warp: True | ||
2098 | Welcome Back Area: | 2206 | Welcome Back Area: |
2099 | door: Welcome Back | 2207 | door: Welcome Back |
2100 | Art Gallery: | ||
2101 | room: Art Gallery | ||
2102 | door: Exit | ||
2103 | The Bearer: | ||
2104 | room: Art Gallery | ||
2105 | door: Exit | ||
2106 | Outside The Initiated: | 2208 | Outside The Initiated: |
2107 | room: Art Gallery | 2209 | room: Art Gallery |
2108 | door: Exit | 2210 | door: Exit |
2211 | warp: True | ||
2109 | panels: | 2212 | panels: |
2110 | SIZE (Small): | 2213 | SIZE (Small): |
2111 | id: Entry Room/Panel_size_small | 2214 | id: Entry Room/Panel_size_small |
@@ -2185,6 +2288,7 @@ | |||
2185 | Orange Tower Fifth Floor: | 2288 | Orange Tower Fifth Floor: |
2186 | room: Orange Tower | 2289 | room: Orange Tower |
2187 | door: Sixth Floor | 2290 | door: Sixth Floor |
2291 | warp: True | ||
2188 | The Scientific: | 2292 | The Scientific: |
2189 | painting: True | 2293 | painting: True |
2190 | paintings: | 2294 | paintings: |
@@ -2213,6 +2317,7 @@ | |||
2213 | Orange Tower Sixth Floor: | 2317 | Orange Tower Sixth Floor: |
2214 | room: Orange Tower | 2318 | room: Orange Tower |
2215 | door: Seventh Floor | 2319 | door: Seventh Floor |
2320 | warp: True | ||
2216 | panels: | 2321 | panels: |
2217 | THE END: | 2322 | THE END: |
2218 | id: EndPanel/Panel_end_end | 2323 | id: EndPanel/Panel_end_end |
@@ -2389,7 +2494,10 @@ | |||
2389 | Courtyard: | 2494 | Courtyard: |
2390 | entrances: | 2495 | entrances: |
2391 | Roof: True | 2496 | Roof: True |
2392 | Orange Tower Fourth Floor: True | 2497 | Orange Tower Fourth Floor: |
2498 | - warp: True | ||
2499 | - room: Crossroads | ||
2500 | door: Tower Entrance | ||
2393 | Arrow Garden: | 2501 | Arrow Garden: |
2394 | painting: True | 2502 | painting: True |
2395 | Starting Room: | 2503 | Starting Room: |
@@ -2757,15 +2865,24 @@ | |||
2757 | entrances: | 2865 | entrances: |
2758 | Starting Room: | 2866 | Starting Room: |
2759 | door: Shortcut to Starting Room | 2867 | door: Shortcut to Starting Room |
2760 | Hub Room: True | 2868 | Hub Room: |
2761 | Outside The Wondrous: True | 2869 | warp: True |
2762 | Outside The Undeterred: True | 2870 | Outside The Wondrous: |
2763 | Outside The Agreeable: True | 2871 | warp: True |
2764 | Outside The Wanderer: True | 2872 | Outside The Undeterred: |
2765 | The Observant: True | 2873 | warp: True |
2766 | Art Gallery: True | 2874 | Outside The Agreeable: |
2767 | The Scientific: True | 2875 | warp: True |
2768 | Cellar: True | 2876 | Outside The Wanderer: |
2877 | warp: True | ||
2878 | The Observant: | ||
2879 | warp: True | ||
2880 | Art Gallery: | ||
2881 | warp: True | ||
2882 | The Scientific: | ||
2883 | warp: True | ||
2884 | Cellar: | ||
2885 | warp: True | ||
2769 | Orange Tower Fifth Floor: | 2886 | Orange Tower Fifth Floor: |
2770 | room: Orange Tower Fifth Floor | 2887 | room: Orange Tower Fifth Floor |
2771 | door: Welcome Back | 2888 | door: Welcome Back |
@@ -2833,10 +2950,21 @@ | |||
2833 | Knight Night Exit: | 2950 | Knight Night Exit: |
2834 | room: Knight Night (Final) | 2951 | room: Knight Night (Final) |
2835 | door: Exit | 2952 | door: Exit |
2836 | Orange Tower Third Floor: True # sunwarp | 2953 | Orange Tower Third Floor: |
2954 | room: Sunwarps | ||
2955 | door: 3 Sunwarp | ||
2956 | sunwarp: True | ||
2837 | Orange Tower Fifth Floor: | 2957 | Orange Tower Fifth Floor: |
2838 | room: Art Gallery | 2958 | room: Art Gallery |
2839 | door: Exit | 2959 | door: Exit |
2960 | warp: True | ||
2961 | Art Gallery: | ||
2962 | room: Art Gallery | ||
2963 | door: Exit | ||
2964 | warp: True | ||
2965 | The Bearer: | ||
2966 | room: Art Gallery | ||
2967 | door: Exit | ||
2840 | Eight Alcove: | 2968 | Eight Alcove: |
2841 | door: Eight Door | 2969 | door: Eight Door |
2842 | The Optimistic: True | 2970 | The Optimistic: True |
@@ -3007,6 +3135,11 @@ | |||
3007 | orientation: east | 3135 | orientation: east |
3008 | - id: smile_painting_1 | 3136 | - id: smile_painting_1 |
3009 | orientation: north | 3137 | orientation: north |
3138 | sunwarps: | ||
3139 | - dots: 3 | ||
3140 | direction: exit | ||
3141 | entrance_indicator_pos: [ 89.99, 2.5, 1 ] | ||
3142 | orientation: east | ||
3010 | The Initiated: | 3143 | The Initiated: |
3011 | entrances: | 3144 | entrances: |
3012 | Outside The Initiated: | 3145 | Outside The Initiated: |
@@ -3130,6 +3263,7 @@ | |||
3130 | door: Traveled Entrance | 3263 | door: Traveled Entrance |
3131 | Color Hallways: | 3264 | Color Hallways: |
3132 | door: Color Hallways Entrance | 3265 | door: Color Hallways Entrance |
3266 | warp: True | ||
3133 | panels: | 3267 | panels: |
3134 | Achievement: | 3268 | Achievement: |
3135 | id: Countdown Panels/Panel_traveled_traveled | 3269 | id: Countdown Panels/Panel_traveled_traveled |
@@ -3220,22 +3354,32 @@ | |||
3220 | The Traveled: | 3354 | The Traveled: |
3221 | room: The Traveled | 3355 | room: The Traveled |
3222 | door: Color Hallways Entrance | 3356 | door: Color Hallways Entrance |
3223 | Outside The Bold: True | 3357 | Outside The Bold: |
3224 | Outside The Undeterred: True | 3358 | warp: True |
3225 | Crossroads: True | 3359 | Outside The Undeterred: |
3226 | Hedge Maze: True | 3360 | warp: True |
3227 | The Optimistic: True # backside | 3361 | Crossroads: |
3228 | Directional Gallery: True # backside | 3362 | warp: True |
3229 | Yellow Backside Area: True | 3363 | Hedge Maze: |
3364 | warp: True | ||
3365 | The Optimistic: | ||
3366 | warp: True # backside | ||
3367 | Directional Gallery: | ||
3368 | warp: True # backside | ||
3369 | Yellow Backside Area: | ||
3370 | warp: True | ||
3230 | The Bearer: | 3371 | The Bearer: |
3231 | room: The Bearer | 3372 | room: The Bearer |
3232 | door: Backside Door | 3373 | door: Backside Door |
3374 | warp: True | ||
3233 | The Observant: | 3375 | The Observant: |
3234 | room: The Observant | 3376 | room: The Observant |
3235 | door: Backside Door | 3377 | door: Backside Door |
3378 | warp: True | ||
3236 | Outside The Bold: | 3379 | Outside The Bold: |
3237 | entrances: | 3380 | entrances: |
3238 | Color Hallways: True | 3381 | Color Hallways: |
3382 | warp: True | ||
3239 | Color Hunt: | 3383 | Color Hunt: |
3240 | room: Color Hunt | 3384 | room: Color Hunt |
3241 | door: Shortcut to The Steady | 3385 | door: Shortcut to The Steady |
@@ -3253,7 +3397,7 @@ | |||
3253 | door: Painting Shortcut | 3397 | door: Painting Shortcut |
3254 | painting: True | 3398 | painting: True |
3255 | Room Room: True # trapdoor | 3399 | Room Room: True # trapdoor |
3256 | Outside The Agreeable: | 3400 | Compass Room: |
3257 | painting: True | 3401 | painting: True |
3258 | panels: | 3402 | panels: |
3259 | UNOPEN: | 3403 | UNOPEN: |
@@ -3455,13 +3599,22 @@ | |||
3455 | tag: botred | 3599 | tag: botred |
3456 | Outside The Undeterred: | 3600 | Outside The Undeterred: |
3457 | entrances: | 3601 | entrances: |
3458 | Color Hallways: True | 3602 | Color Hallways: |
3459 | Orange Tower First Floor: True # sunwarp | 3603 | warp: True |
3460 | Orange Tower Second Floor: True | 3604 | Orange Tower First Floor: |
3461 | The Artistic (Smiley): True | 3605 | room: Sunwarps |
3462 | The Artistic (Panda): True | 3606 | door: 4 Sunwarp |
3463 | The Artistic (Apple): True | 3607 | sunwarp: True |
3464 | The Artistic (Lattice): True | 3608 | Orange Tower Second Floor: |
3609 | warp: True | ||
3610 | The Artistic (Smiley): | ||
3611 | warp: True | ||
3612 | The Artistic (Panda): | ||
3613 | warp: True | ||
3614 | The Artistic (Apple): | ||
3615 | warp: True | ||
3616 | The Artistic (Lattice): | ||
3617 | warp: True | ||
3465 | Yellow Backside Area: | 3618 | Yellow Backside Area: |
3466 | painting: True | 3619 | painting: True |
3467 | Number Hunt: | 3620 | Number Hunt: |
@@ -3651,6 +3804,11 @@ | |||
3651 | door: Green Painting | 3804 | door: Green Painting |
3652 | - id: blueman_painting_2 | 3805 | - id: blueman_painting_2 |
3653 | orientation: east | 3806 | orientation: east |
3807 | sunwarps: | ||
3808 | - dots: 4 | ||
3809 | direction: exit | ||
3810 | entrance_indicator_pos: [ -89.01, 2.5, 4 ] | ||
3811 | orientation: east | ||
3654 | The Undeterred: | 3812 | The Undeterred: |
3655 | entrances: | 3813 | entrances: |
3656 | Outside The Undeterred: | 3814 | Outside The Undeterred: |
@@ -3928,7 +4086,10 @@ | |||
3928 | door: Eights | 4086 | door: Eights |
3929 | Directional Gallery: | 4087 | Directional Gallery: |
3930 | entrances: | 4088 | entrances: |
3931 | Outside The Agreeable: True # sunwarp | 4089 | Outside The Agreeable: |
4090 | room: Sunwarps | ||
4091 | door: 6 Sunwarp | ||
4092 | sunwarp: True | ||
3932 | Orange Tower First Floor: | 4093 | Orange Tower First Floor: |
3933 | room: Orange Tower First Floor | 4094 | room: Orange Tower First Floor |
3934 | door: Salt Pepper Door | 4095 | door: Salt Pepper Door |
@@ -4096,11 +4257,19 @@ | |||
4096 | orientation: south | 4257 | orientation: south |
4097 | - id: cherry_painting | 4258 | - id: cherry_painting |
4098 | orientation: east | 4259 | orientation: east |
4260 | sunwarps: | ||
4261 | - dots: 6 | ||
4262 | direction: exit | ||
4263 | entrance_indicator_pos: [ -39, 2.5, -7.01 ] | ||
4264 | orientation: north | ||
4099 | Color Hunt: | 4265 | Color Hunt: |
4100 | entrances: | 4266 | entrances: |
4101 | Outside The Bold: | 4267 | Outside The Bold: |
4102 | door: Shortcut to The Steady | 4268 | door: Shortcut to The Steady |
4103 | Orange Tower Fourth Floor: True # sunwarp | 4269 | Orange Tower Fourth Floor: |
4270 | room: Sunwarps | ||
4271 | door: 5 Sunwarp | ||
4272 | sunwarp: True | ||
4104 | Roof: True # through ceiling of sunwarp | 4273 | Roof: True # through ceiling of sunwarp |
4105 | Champion's Rest: | 4274 | Champion's Rest: |
4106 | room: Outside The Initiated | 4275 | room: Outside The Initiated |
@@ -4159,6 +4328,11 @@ | |||
4159 | required_door: | 4328 | required_door: |
4160 | room: Outside The Initiated | 4329 | room: Outside The Initiated |
4161 | door: Entrance | 4330 | door: Entrance |
4331 | sunwarps: | ||
4332 | - dots: 5 | ||
4333 | direction: exit | ||
4334 | entrance_indicator_pos: [ 54, 2.5, 69.99 ] | ||
4335 | orientation: north | ||
4162 | Champion's Rest: | 4336 | Champion's Rest: |
4163 | entrances: | 4337 | entrances: |
4164 | Color Hunt: | 4338 | Color Hunt: |
@@ -4192,7 +4366,7 @@ | |||
4192 | entrances: | 4366 | entrances: |
4193 | Outside The Bold: | 4367 | Outside The Bold: |
4194 | door: Entrance | 4368 | door: Entrance |
4195 | Orange Tower Fifth Floor: | 4369 | Outside The Initiated: |
4196 | room: Art Gallery | 4370 | room: Art Gallery |
4197 | door: Exit | 4371 | door: Exit |
4198 | The Bearer (East): True | 4372 | The Bearer (East): True |
@@ -4640,7 +4814,8 @@ | |||
4640 | tag: midyellow | 4814 | tag: midyellow |
4641 | The Steady (Lime): | 4815 | The Steady (Lime): |
4642 | entrances: | 4816 | entrances: |
4643 | The Steady (Sunflower): True | 4817 | The Steady (Sunflower): |
4818 | warp: True | ||
4644 | The Steady (Emerald): | 4819 | The Steady (Emerald): |
4645 | room: The Steady | 4820 | room: The Steady |
4646 | door: Reveal | 4821 | door: Reveal |
@@ -4662,7 +4837,8 @@ | |||
4662 | orientation: south | 4837 | orientation: south |
4663 | The Steady (Lemon): | 4838 | The Steady (Lemon): |
4664 | entrances: | 4839 | entrances: |
4665 | The Steady (Emerald): True | 4840 | The Steady (Emerald): |
4841 | warp: True | ||
4666 | The Steady (Orange): | 4842 | The Steady (Orange): |
4667 | room: The Steady | 4843 | room: The Steady |
4668 | door: Reveal | 4844 | door: Reveal |
@@ -5019,8 +5195,10 @@ | |||
5019 | Knight Night (Outer Ring): | 5195 | Knight Night (Outer Ring): |
5020 | room: Knight Night (Outer Ring) | 5196 | room: Knight Night (Outer Ring) |
5021 | door: Fore Door | 5197 | door: Fore Door |
5198 | warp: True | ||
5022 | Knight Night (Right Lower Segment): | 5199 | Knight Night (Right Lower Segment): |
5023 | door: Segment Door | 5200 | door: Segment Door |
5201 | warp: True | ||
5024 | panels: | 5202 | panels: |
5025 | RUST (1): | 5203 | RUST (1): |
5026 | id: Appendix Room/Panel_rust_trust | 5204 | id: Appendix Room/Panel_rust_trust |
@@ -5049,9 +5227,11 @@ | |||
5049 | Knight Night (Right Upper Segment): | 5227 | Knight Night (Right Upper Segment): |
5050 | room: Knight Night (Right Upper Segment) | 5228 | room: Knight Night (Right Upper Segment) |
5051 | door: Segment Door | 5229 | door: Segment Door |
5230 | warp: True | ||
5052 | Knight Night (Outer Ring): | 5231 | Knight Night (Outer Ring): |
5053 | room: Knight Night (Outer Ring) | 5232 | room: Knight Night (Outer Ring) |
5054 | door: New Door | 5233 | door: New Door |
5234 | warp: True | ||
5055 | panels: | 5235 | panels: |
5056 | ADJUST: | 5236 | ADJUST: |
5057 | id: Appendix Room/Panel_adjust_readjusted | 5237 | id: Appendix Room/Panel_adjust_readjusted |
@@ -5097,9 +5277,11 @@ | |||
5097 | Knight Night (Outer Ring): | 5277 | Knight Night (Outer Ring): |
5098 | room: Knight Night (Outer Ring) | 5278 | room: Knight Night (Outer Ring) |
5099 | door: To End | 5279 | door: To End |
5280 | warp: True | ||
5100 | Knight Night (Right Upper Segment): | 5281 | Knight Night (Right Upper Segment): |
5101 | room: Knight Night (Outer Ring) | 5282 | room: Knight Night (Outer Ring) |
5102 | door: To End | 5283 | door: To End |
5284 | warp: True | ||
5103 | panels: | 5285 | panels: |
5104 | TRUSTED: | 5286 | TRUSTED: |
5105 | id: Appendix Room/Panel_trusted_readjusted | 5287 | id: Appendix Room/Panel_trusted_readjusted |
@@ -5295,7 +5477,7 @@ | |||
5295 | entrances: | 5477 | entrances: |
5296 | Orange Tower Sixth Floor: | 5478 | Orange Tower Sixth Floor: |
5297 | painting: True | 5479 | painting: True |
5298 | Outside The Agreeable: | 5480 | Hallway Room (1): |
5299 | painting: True | 5481 | painting: True |
5300 | The Artistic (Smiley): | 5482 | The Artistic (Smiley): |
5301 | room: The Artistic (Smiley) | 5483 | room: The Artistic (Smiley) |
@@ -5746,7 +5928,8 @@ | |||
5746 | painting: True | 5928 | painting: True |
5747 | Wondrous Lobby: | 5929 | Wondrous Lobby: |
5748 | door: Exit | 5930 | door: Exit |
5749 | Directional Gallery: True | 5931 | Directional Gallery: |
5932 | warp: True | ||
5750 | panels: | 5933 | panels: |
5751 | NEAR: | 5934 | NEAR: |
5752 | id: Shuffle Room/Panel_near_near | 5935 | id: Shuffle Room/Panel_near_near |
@@ -5781,7 +5964,8 @@ | |||
5781 | tag: midwhite | 5964 | tag: midwhite |
5782 | Wondrous Lobby: | 5965 | Wondrous Lobby: |
5783 | entrances: | 5966 | entrances: |
5784 | Directional Gallery: True | 5967 | Directional Gallery: |
5968 | warp: True | ||
5785 | The Eyes They See: | 5969 | The Eyes They See: |
5786 | room: The Eyes They See | 5970 | room: The Eyes They See |
5787 | door: Exit | 5971 | door: Exit |
@@ -5790,10 +5974,12 @@ | |||
5790 | orientation: east | 5974 | orientation: east |
5791 | Outside The Wondrous: | 5975 | Outside The Wondrous: |
5792 | entrances: | 5976 | entrances: |
5793 | Wondrous Lobby: True | 5977 | Wondrous Lobby: |
5978 | warp: True | ||
5794 | The Wondrous (Doorknob): | 5979 | The Wondrous (Doorknob): |
5795 | door: Wondrous Entrance | 5980 | door: Wondrous Entrance |
5796 | The Wondrous (Window): True | 5981 | The Wondrous (Window): |
5982 | warp: True | ||
5797 | panels: | 5983 | panels: |
5798 | SHRINK: | 5984 | SHRINK: |
5799 | id: Wonderland Room/Panel_shrink_shrink | 5985 | id: Wonderland Room/Panel_shrink_shrink |
@@ -5815,7 +6001,9 @@ | |||
5815 | painting: True | 6001 | painting: True |
5816 | The Wondrous (Chandelier): | 6002 | The Wondrous (Chandelier): |
5817 | painting: True | 6003 | painting: True |
5818 | The Wondrous (Table): True # There is a way that doesn't use the painting | 6004 | The Wondrous (Table): |
6005 | - painting: True | ||
6006 | - warp: True | ||
5819 | doors: | 6007 | doors: |
5820 | Painting Shortcut: | 6008 | Painting Shortcut: |
5821 | painting_id: | 6009 | painting_id: |
@@ -5901,7 +6089,8 @@ | |||
5901 | required: True | 6089 | required: True |
5902 | The Wondrous: | 6090 | The Wondrous: |
5903 | entrances: | 6091 | entrances: |
5904 | The Wondrous (Table): True | 6092 | The Wondrous (Table): |
6093 | warp: True | ||
5905 | Arrow Garden: | 6094 | Arrow Garden: |
5906 | door: Exit | 6095 | door: Exit |
5907 | panels: | 6096 | panels: |
@@ -5967,11 +6156,70 @@ | |||
5967 | paintings: | 6156 | paintings: |
5968 | - id: flower_painting_6 | 6157 | - id: flower_painting_6 |
5969 | orientation: south | 6158 | orientation: south |
5970 | Hallway Room (2): | 6159 | Hallway Room (1): |
5971 | entrances: | 6160 | entrances: |
5972 | Outside The Agreeable: | 6161 | Outside The Agreeable: |
5973 | room: Outside The Agreeable | 6162 | warp: True |
5974 | door: Hallway Door | 6163 | Hallway Room (2): |
6164 | warp: True | ||
6165 | Hallway Room (3): | ||
6166 | warp: True | ||
6167 | Hallway Room (4): | ||
6168 | warp: True | ||
6169 | panels: | ||
6170 | OUT: | ||
6171 | id: Hallway Room/Panel_out_out | ||
6172 | check: True | ||
6173 | exclude_reduce: True | ||
6174 | tag: midwhite | ||
6175 | WALL: | ||
6176 | id: Hallway Room/Panel_castle_1 | ||
6177 | colors: blue | ||
6178 | tag: quad bot blue | ||
6179 | link: qbb CASTLE | ||
6180 | KEEP: | ||
6181 | id: Hallway Room/Panel_castle_2 | ||
6182 | colors: blue | ||
6183 | tag: quad bot blue | ||
6184 | link: qbb CASTLE | ||
6185 | BAILEY: | ||
6186 | id: Hallway Room/Panel_castle_3 | ||
6187 | colors: blue | ||
6188 | tag: quad bot blue | ||
6189 | link: qbb CASTLE | ||
6190 | TOWER: | ||
6191 | id: Hallway Room/Panel_castle_4 | ||
6192 | colors: blue | ||
6193 | tag: quad bot blue | ||
6194 | link: qbb CASTLE | ||
6195 | doors: | ||
6196 | Exit: | ||
6197 | id: Red Blue Purple Room Area Doors/Door_room_2 | ||
6198 | door_group: Hallway Room Doors | ||
6199 | location_name: Hallway Room - First Room | ||
6200 | panels: | ||
6201 | - WALL | ||
6202 | - KEEP | ||
6203 | - BAILEY | ||
6204 | - TOWER | ||
6205 | paintings: | ||
6206 | - id: panda_painting | ||
6207 | orientation: south | ||
6208 | progression: | ||
6209 | Progressive Hallway Room: | ||
6210 | - Exit | ||
6211 | - room: Hallway Room (2) | ||
6212 | door: Exit | ||
6213 | - room: Hallway Room (3) | ||
6214 | door: Exit | ||
6215 | - room: Hallway Room (4) | ||
6216 | door: Exit | ||
6217 | Hallway Room (2): | ||
6218 | entrances: | ||
6219 | Hallway Room (1): | ||
6220 | room: Hallway Room (1) | ||
6221 | door: Exit | ||
6222 | warp: True | ||
5975 | Elements Area: True | 6223 | Elements Area: True |
5976 | panels: | 6224 | panels: |
5977 | WISE: | 6225 | WISE: |
@@ -6009,6 +6257,7 @@ | |||
6009 | Hallway Room (2): | 6257 | Hallway Room (2): |
6010 | room: Hallway Room (2) | 6258 | room: Hallway Room (2) |
6011 | door: Exit | 6259 | door: Exit |
6260 | warp: True | ||
6012 | # No entrance from Elements Area. The winding hallway does not connect. | 6261 | # No entrance from Elements Area. The winding hallway does not connect. |
6013 | panels: | 6262 | panels: |
6014 | TRANCE: | 6263 | TRANCE: |
@@ -6046,6 +6295,7 @@ | |||
6046 | Hallway Room (3): | 6295 | Hallway Room (3): |
6047 | room: Hallway Room (3) | 6296 | room: Hallway Room (3) |
6048 | door: Exit | 6297 | door: Exit |
6298 | warp: True | ||
6049 | Elements Area: True | 6299 | Elements Area: True |
6050 | panels: | 6300 | panels: |
6051 | WHEEL: | 6301 | WHEEL: |
@@ -6068,6 +6318,7 @@ | |||
6068 | Hallway Room (4): | 6318 | Hallway Room (4): |
6069 | room: Hallway Room (4) | 6319 | room: Hallway Room (4) |
6070 | door: Exit | 6320 | door: Exit |
6321 | # If this door is open, then a non-warp entrance from the first hallway room is available | ||
6071 | The Artistic (Smiley): | 6322 | The Artistic (Smiley): |
6072 | room: Hallway Room (4) | 6323 | room: Hallway Room (4) |
6073 | door: Exit | 6324 | door: Exit |
@@ -6112,6 +6363,7 @@ | |||
6112 | entrances: | 6363 | entrances: |
6113 | Orange Tower First Floor: | 6364 | Orange Tower First Floor: |
6114 | door: Tower Entrance | 6365 | door: Tower Entrance |
6366 | warp: True | ||
6115 | Rhyme Room (Cross): | 6367 | Rhyme Room (Cross): |
6116 | room: Rhyme Room (Cross) | 6368 | room: Rhyme Room (Cross) |
6117 | door: Exit | 6369 | door: Exit |
@@ -6139,6 +6391,7 @@ | |||
6139 | Outside The Wanderer: | 6391 | Outside The Wanderer: |
6140 | room: Outside The Wanderer | 6392 | room: Outside The Wanderer |
6141 | door: Wanderer Entrance | 6393 | door: Wanderer Entrance |
6394 | warp: True | ||
6142 | panels: | 6395 | panels: |
6143 | Achievement: | 6396 | Achievement: |
6144 | id: Countdown Panels/Panel_1234567890_wanderlust | 6397 | id: Countdown Panels/Panel_1234567890_wanderlust |
@@ -6180,12 +6433,17 @@ | |||
6180 | tag: midorange | 6433 | tag: midorange |
6181 | Art Gallery: | 6434 | Art Gallery: |
6182 | entrances: | 6435 | entrances: |
6183 | Orange Tower Third Floor: True | 6436 | Orange Tower Third Floor: |
6184 | Art Gallery (Second Floor): True | 6437 | warp: True |
6185 | Art Gallery (Third Floor): True | 6438 | Art Gallery (Second Floor): |
6186 | Art Gallery (Fourth Floor): True | 6439 | warp: True |
6187 | Orange Tower Fifth Floor: | 6440 | Art Gallery (Third Floor): |
6441 | warp: True | ||
6442 | Art Gallery (Fourth Floor): | ||
6443 | warp: True | ||
6444 | Outside The Initiated: | ||
6188 | door: Exit | 6445 | door: Exit |
6446 | warp: True | ||
6189 | panels: | 6447 | panels: |
6190 | EIGHT: | 6448 | EIGHT: |
6191 | id: Backside Room/Panel_eight_eight_6 | 6449 | id: Backside Room/Panel_eight_eight_6 |
@@ -6766,6 +7024,7 @@ | |||
6766 | Rhyme Room (Smiley): # one-way | 7024 | Rhyme Room (Smiley): # one-way |
6767 | room: Rhyme Room (Smiley) | 7025 | room: Rhyme Room (Smiley) |
6768 | door: Door to Target | 7026 | door: Door to Target |
7027 | warp: True | ||
6769 | Rhyme Room (Looped Square): | 7028 | Rhyme Room (Looped Square): |
6770 | room: Rhyme Room (Looped Square) | 7029 | room: Rhyme Room (Looped Square) |
6771 | door: Door to Target | 7030 | door: Door to Target |
@@ -6829,7 +7088,8 @@ | |||
6829 | # For pretty much the same reason, I don't want to shuffle the paintings in | 7088 | # For pretty much the same reason, I don't want to shuffle the paintings in |
6830 | # here. | 7089 | # here. |
6831 | entrances: | 7090 | entrances: |
6832 | Orange Tower Fourth Floor: True | 7091 | Orange Tower Fourth Floor: |
7092 | warp: True | ||
6833 | panels: | 7093 | panels: |
6834 | DOOR (1): | 7094 | DOOR (1): |
6835 | id: Panel Room/Panel_room_door_1 | 7095 | id: Panel Room/Panel_room_door_1 |
@@ -7037,9 +7297,11 @@ | |||
7037 | Orange Tower Fifth Floor: | 7297 | Orange Tower Fifth Floor: |
7038 | room: Room Room | 7298 | room: Room Room |
7039 | door: Cellar Exit | 7299 | door: Cellar Exit |
7040 | Outside The Agreeable: | 7300 | warp: True |
7041 | room: Outside The Agreeable | 7301 | Compass Room: |
7302 | room: Compass Room | ||
7042 | door: Lookout Entrance | 7303 | door: Lookout Entrance |
7304 | warp: True | ||
7043 | Outside The Wise: | 7305 | Outside The Wise: |
7044 | entrances: | 7306 | entrances: |
7045 | Orange Tower Sixth Floor: | 7307 | Orange Tower Sixth Floor: |
@@ -7077,6 +7339,7 @@ | |||
7077 | Outside The Wise: | 7339 | Outside The Wise: |
7078 | room: Outside The Wise | 7340 | room: Outside The Wise |
7079 | door: Wise Entrance | 7341 | door: Wise Entrance |
7342 | warp: True # The Wise is so full of warps | ||
7080 | panels: | 7343 | panels: |
7081 | Achievement: | 7344 | Achievement: |
7082 | id: Countdown Panels/Panel_intelligent_wise | 7345 | id: Countdown Panels/Panel_intelligent_wise |
diff --git a/data/generated.dat b/data/generated.dat index c957e5d..304109c 100644 --- a/data/generated.dat +++ b/data/generated.dat | |||
Binary files differ | |||
diff --git a/data/ids.yaml b/data/ids.yaml index d3307de..918af7a 100644 --- a/data/ids.yaml +++ b/data/ids.yaml | |||
@@ -140,13 +140,9 @@ panels: | |||
140 | PURPLE: 444502 | 140 | PURPLE: 444502 |
141 | FIVE (1): 444503 | 141 | FIVE (1): 444503 |
142 | FIVE (2): 444504 | 142 | FIVE (2): 444504 |
143 | OUT: 444505 | ||
144 | HIDE: 444506 | 143 | HIDE: 444506 |
145 | DAZE: 444507 | 144 | DAZE: 444507 |
146 | WALL: 444508 | 145 | Compass Room: |
147 | KEEP: 444509 | ||
148 | BAILEY: 444510 | ||
149 | TOWER: 444511 | ||
150 | NORTH: 444512 | 146 | NORTH: 444512 |
151 | DIAMONDS: 444513 | 147 | DIAMONDS: 444513 |
152 | FIRE: 444514 | 148 | FIRE: 444514 |
@@ -689,6 +685,12 @@ panels: | |||
689 | Arrow Garden: | 685 | Arrow Garden: |
690 | MASTERY: 444948 | 686 | MASTERY: 444948 |
691 | SHARP: 444949 | 687 | SHARP: 444949 |
688 | Hallway Room (1): | ||
689 | OUT: 444505 | ||
690 | WALL: 444508 | ||
691 | KEEP: 444509 | ||
692 | BAILEY: 444510 | ||
693 | TOWER: 444511 | ||
692 | Hallway Room (2): | 694 | Hallway Room (2): |
693 | WISE: 444950 | 695 | WISE: 444950 |
694 | CLOCK: 444951 | 696 | CLOCK: 444951 |
@@ -995,6 +997,19 @@ doors: | |||
995 | Traveled Entrance: | 997 | Traveled Entrance: |
996 | item: 444433 | 998 | item: 444433 |
997 | location: 444438 | 999 | location: 444438 |
1000 | Sunwarps: | ||
1001 | 1 Sunwarp: | ||
1002 | item: 444581 | ||
1003 | 2 Sunwarp: | ||
1004 | item: 444588 | ||
1005 | 3 Sunwarp: | ||
1006 | item: 444586 | ||
1007 | 4 Sunwarp: | ||
1008 | item: 444585 | ||
1009 | 5 Sunwarp: | ||
1010 | item: 444587 | ||
1011 | 6 Sunwarp: | ||
1012 | item: 444584 | ||
998 | Pilgrim Antechamber: | 1013 | Pilgrim Antechamber: |
999 | Sun Painting: | 1014 | Sun Painting: |
1000 | item: 444436 | 1015 | item: 444436 |
@@ -1067,9 +1082,7 @@ doors: | |||
1067 | location: 444501 | 1082 | location: 444501 |
1068 | Purple Barrier: | 1083 | Purple Barrier: |
1069 | item: 444457 | 1084 | item: 444457 |
1070 | Hallway Door: | 1085 | Compass Room: |
1071 | item: 444459 | ||
1072 | location: 445214 | ||
1073 | Lookout Entrance: | 1086 | Lookout Entrance: |
1074 | item: 444579 | 1087 | item: 444579 |
1075 | location: 445271 | 1088 | location: 445271 |
@@ -1342,6 +1355,10 @@ doors: | |||
1342 | Exit: | 1355 | Exit: |
1343 | item: 444552 | 1356 | item: 444552 |
1344 | location: 444947 | 1357 | location: 444947 |
1358 | Hallway Room (1): | ||
1359 | Exit: | ||
1360 | item: 444459 | ||
1361 | location: 445214 | ||
1345 | Hallway Room (2): | 1362 | Hallway Room (2): |
1346 | Exit: | 1363 | Exit: |
1347 | item: 444553 | 1364 | item: 444553 |
@@ -1452,9 +1469,11 @@ door_groups: | |||
1452 | Colorful Doors: 444498 | 1469 | Colorful Doors: 444498 |
1453 | Directional Gallery Doors: 444531 | 1470 | Directional Gallery Doors: 444531 |
1454 | Artistic Doors: 444545 | 1471 | Artistic Doors: 444545 |
1472 | Sunwarps: 444582 | ||
1455 | progression: | 1473 | progression: |
1456 | Progressive Hallway Room: 444461 | 1474 | Progressive Hallway Room: 444461 |
1457 | Progressive Fearless: 444470 | 1475 | Progressive Fearless: 444470 |
1458 | Progressive Orange Tower: 444482 | 1476 | Progressive Orange Tower: 444482 |
1459 | Progressive Art Gallery: 444563 | 1477 | Progressive Art Gallery: 444563 |
1460 | Progressive Colorful: 444580 | 1478 | Progressive Colorful: 444580 |
1479 | Progressive Pilgrimage: 444583 | ||
diff --git a/datatypes.py b/datatypes.py index e9bf0a3..e466558 100644 --- a/datatypes.py +++ b/datatypes.py | |||
@@ -1,3 +1,4 @@ | |||
1 | from enum import Enum, Flag, auto | ||
1 | from typing import List, NamedTuple, Optional | 2 | from typing import List, NamedTuple, Optional |
2 | 3 | ||
3 | 4 | ||
@@ -11,10 +12,18 @@ class RoomAndPanel(NamedTuple): | |||
11 | panel: str | 12 | panel: str |
12 | 13 | ||
13 | 14 | ||
15 | class EntranceType(Flag): | ||
16 | NORMAL = auto() | ||
17 | PAINTING = auto() | ||
18 | SUNWARP = auto() | ||
19 | WARP = auto() | ||
20 | CROSSROADS_ROOF_ACCESS = auto() | ||
21 | |||
22 | |||
14 | class RoomEntrance(NamedTuple): | 23 | class RoomEntrance(NamedTuple): |
15 | room: str # source room | 24 | room: str # source room |
16 | door: Optional[RoomAndDoor] | 25 | door: Optional[RoomAndDoor] |
17 | painting: bool | 26 | type: EntranceType |
18 | 27 | ||
19 | 28 | ||
20 | class Room(NamedTuple): | 29 | class Room(NamedTuple): |
@@ -22,6 +31,12 @@ class Room(NamedTuple): | |||
22 | entrances: List[RoomEntrance] | 31 | entrances: List[RoomEntrance] |
23 | 32 | ||
24 | 33 | ||
34 | class DoorType(Enum): | ||
35 | NORMAL = 1 | ||
36 | SUNWARP = 2 | ||
37 | SUN_PAINTING = 3 | ||
38 | |||
39 | |||
25 | class Door(NamedTuple): | 40 | class Door(NamedTuple): |
26 | name: str | 41 | name: str |
27 | item_name: str | 42 | item_name: str |
@@ -34,7 +49,7 @@ class Door(NamedTuple): | |||
34 | event: bool | 49 | event: bool |
35 | door_group: Optional[str] | 50 | door_group: Optional[str] |
36 | include_reduce: bool | 51 | include_reduce: bool |
37 | junk_item: bool | 52 | type: DoorType |
38 | item_group: Optional[str] | 53 | item_group: Optional[str] |
39 | 54 | ||
40 | 55 | ||
diff --git a/items.py b/items.py index 7c7928c..67eacea 100644 --- a/items.py +++ b/items.py | |||
@@ -1,8 +1,14 @@ | |||
1 | from typing import Dict, List, NamedTuple, Optional, TYPE_CHECKING | 1 | from enum import Enum |
2 | from typing import Dict, List, NamedTuple, Set | ||
2 | 3 | ||
3 | from BaseClasses import Item, ItemClassification | 4 | from BaseClasses import Item, ItemClassification |
4 | from .static_logic import DOORS_BY_ROOM, PROGRESSION_BY_ROOM, PROGRESSIVE_ITEMS, get_door_group_item_id, \ | 5 | from .static_logic import DOORS_BY_ROOM, PROGRESSIVE_ITEMS, get_door_group_item_id, get_door_item_id, \ |
5 | get_door_item_id, get_progressive_item_id, get_special_item_id | 6 | get_progressive_item_id, get_special_item_id |
7 | |||
8 | |||
9 | class ItemType(Enum): | ||
10 | NORMAL = 1 | ||
11 | COLOR = 2 | ||
6 | 12 | ||
7 | 13 | ||
8 | class ItemData(NamedTuple): | 14 | class ItemData(NamedTuple): |
@@ -11,7 +17,7 @@ class ItemData(NamedTuple): | |||
11 | """ | 17 | """ |
12 | code: int | 18 | code: int |
13 | classification: ItemClassification | 19 | classification: ItemClassification |
14 | mode: Optional[str] | 20 | type: ItemType |
15 | has_doors: bool | 21 | has_doors: bool |
16 | painting_ids: List[str] | 22 | painting_ids: List[str] |
17 | 23 | ||
@@ -34,36 +40,29 @@ def load_item_data(): | |||
34 | 40 | ||
35 | for color in ["Black", "Red", "Blue", "Yellow", "Green", "Orange", "Gray", "Brown", "Purple"]: | 41 | for color in ["Black", "Red", "Blue", "Yellow", "Green", "Orange", "Gray", "Brown", "Purple"]: |
36 | ALL_ITEM_TABLE[color] = ItemData(get_special_item_id(color), ItemClassification.progression, | 42 | ALL_ITEM_TABLE[color] = ItemData(get_special_item_id(color), ItemClassification.progression, |
37 | "colors", [], []) | 43 | ItemType.COLOR, False, []) |
38 | ITEMS_BY_GROUP.setdefault("Colors", []).append(color) | 44 | ITEMS_BY_GROUP.setdefault("Colors", []).append(color) |
39 | 45 | ||
40 | door_groups: Dict[str, List[str]] = {} | 46 | door_groups: Set[str] = set() |
41 | for room_name, doors in DOORS_BY_ROOM.items(): | 47 | for room_name, doors in DOORS_BY_ROOM.items(): |
42 | for door_name, door in doors.items(): | 48 | for door_name, door in doors.items(): |
43 | if door.skip_item is True or door.event is True: | 49 | if door.skip_item is True or door.event is True: |
44 | continue | 50 | continue |
45 | 51 | ||
46 | if door.door_group is None: | 52 | if door.door_group is not None: |
47 | door_mode = "doors" | 53 | door_groups.add(door.door_group) |
48 | else: | ||
49 | door_mode = "complex door" | ||
50 | door_groups.setdefault(door.door_group, []) | ||
51 | |||
52 | if room_name in PROGRESSION_BY_ROOM and door_name in PROGRESSION_BY_ROOM[room_name]: | ||
53 | door_mode = "special" | ||
54 | 54 | ||
55 | ALL_ITEM_TABLE[door.item_name] = \ | 55 | ALL_ITEM_TABLE[door.item_name] = \ |
56 | ItemData(get_door_item_id(room_name, door_name), | 56 | ItemData(get_door_item_id(room_name, door_name), ItemClassification.progression, ItemType.NORMAL, |
57 | ItemClassification.filler if door.junk_item else ItemClassification.progression, door_mode, | ||
58 | door.has_doors, door.painting_ids) | 57 | door.has_doors, door.painting_ids) |
59 | ITEMS_BY_GROUP.setdefault("Doors", []).append(door.item_name) | 58 | ITEMS_BY_GROUP.setdefault("Doors", []).append(door.item_name) |
60 | 59 | ||
61 | if door.item_group is not None: | 60 | if door.item_group is not None: |
62 | ITEMS_BY_GROUP.setdefault(door.item_group, []).append(door.item_name) | 61 | ITEMS_BY_GROUP.setdefault(door.item_group, []).append(door.item_name) |
63 | 62 | ||
64 | for group, group_door_ids in door_groups.items(): | 63 | for group in door_groups: |
65 | ALL_ITEM_TABLE[group] = ItemData(get_door_group_item_id(group), | 64 | ALL_ITEM_TABLE[group] = ItemData(get_door_group_item_id(group), |
66 | ItemClassification.progression, "door group", True, []) | 65 | ItemClassification.progression, ItemType.NORMAL, True, []) |
67 | ITEMS_BY_GROUP.setdefault("Doors", []).append(group) | 66 | ITEMS_BY_GROUP.setdefault("Doors", []).append(group) |
68 | 67 | ||
69 | special_items: Dict[str, ItemClassification] = { | 68 | special_items: Dict[str, ItemClassification] = { |
@@ -77,7 +76,7 @@ def load_item_data(): | |||
77 | 76 | ||
78 | for item_name, classification in special_items.items(): | 77 | for item_name, classification in special_items.items(): |
79 | ALL_ITEM_TABLE[item_name] = ItemData(get_special_item_id(item_name), classification, | 78 | ALL_ITEM_TABLE[item_name] = ItemData(get_special_item_id(item_name), classification, |
80 | "special", False, []) | 79 | ItemType.NORMAL, False, []) |
81 | 80 | ||
82 | if classification == ItemClassification.filler: | 81 | if classification == ItemClassification.filler: |
83 | ITEMS_BY_GROUP.setdefault("Junk", []).append(item_name) | 82 | ITEMS_BY_GROUP.setdefault("Junk", []).append(item_name) |
@@ -86,7 +85,7 @@ def load_item_data(): | |||
86 | 85 | ||
87 | for item_name in PROGRESSIVE_ITEMS: | 86 | for item_name in PROGRESSIVE_ITEMS: |
88 | ALL_ITEM_TABLE[item_name] = ItemData(get_progressive_item_id(item_name), | 87 | ALL_ITEM_TABLE[item_name] = ItemData(get_progressive_item_id(item_name), |
89 | ItemClassification.progression, "special", False, []) | 88 | ItemClassification.progression, ItemType.NORMAL, False, []) |
90 | 89 | ||
91 | 90 | ||
92 | # Initialize the item data at module scope. | 91 | # Initialize the item data at module scope. |
diff --git a/locations.py b/locations.py index 92ee309..a6e53e7 100644 --- a/locations.py +++ b/locations.py | |||
@@ -56,7 +56,7 @@ def load_location_data(): | |||
56 | 56 | ||
57 | for room_name, doors in DOORS_BY_ROOM.items(): | 57 | for room_name, doors in DOORS_BY_ROOM.items(): |
58 | for door_name, door in doors.items(): | 58 | for door_name, door in doors.items(): |
59 | if door.skip_location or door.event or door.panels is None: | 59 | if door.skip_location or door.event or not door.panels: |
60 | continue | 60 | continue |
61 | 61 | ||
62 | location_name = door.location_name | 62 | location_name = door.location_name |
diff --git a/options.py b/options.py index 293992a..05fb4ed 100644 --- a/options.py +++ b/options.py | |||
@@ -61,15 +61,55 @@ class ShufflePaintings(Toggle): | |||
61 | display_name = "Shuffle Paintings" | 61 | display_name = "Shuffle Paintings" |
62 | 62 | ||
63 | 63 | ||
64 | class EnablePilgrimage(Toggle): | ||
65 | """If on, you are required to complete a pilgrimage in order to access the Pilgrim Antechamber. | ||
66 | If off, the pilgrimage will be deactivated, and the sun painting will be added to the pool, even if door shuffle is off.""" | ||
67 | display_name = "Enable Pilgrimage" | ||
68 | |||
69 | |||
70 | class PilgrimageAllowsRoofAccess(DefaultOnToggle): | ||
71 | """If on, you may use the Crossroads roof access during a pilgrimage (and you may be expected to do so). | ||
72 | Otherwise, pilgrimage will be deactivated when going up the stairs.""" | ||
73 | display_name = "Allow Roof Access for Pilgrimage" | ||
74 | |||
75 | |||
76 | class PilgrimageAllowsPaintings(DefaultOnToggle): | ||
77 | """If on, you may use paintings during a pilgrimage (and you may be expected to do so). | ||
78 | Otherwise, pilgrimage will be deactivated when going through a painting.""" | ||
79 | display_name = "Allow Paintings for Pilgrimage" | ||
80 | |||
81 | |||
82 | class SunwarpAccess(Choice): | ||
83 | """Determines how access to sunwarps works. | ||
84 | On "normal", all sunwarps are enabled from the start. | ||
85 | On "disabled", all sunwarps are disabled. Pilgrimage must be disabled when this is used. | ||
86 | On "unlock", sunwarps start off disabled, and all six activate once you receive an item. | ||
87 | On "individual", sunwarps start off disabled, and each has a corresponding item that unlocks it. | ||
88 | On "progressive", sunwarps start off disabled, and they unlock in order using a progressive item.""" | ||
89 | display_name = "Sunwarp Access" | ||
90 | option_normal = 0 | ||
91 | option_disabled = 1 | ||
92 | option_unlock = 2 | ||
93 | option_individual = 3 | ||
94 | option_progressive = 4 | ||
95 | |||
96 | |||
97 | class ShuffleSunwarps(Toggle): | ||
98 | """If on, the pairing and ordering of the sunwarps in the game will be randomized.""" | ||
99 | display_name = "Shuffle Sunwarps" | ||
100 | |||
101 | |||
64 | class VictoryCondition(Choice): | 102 | class VictoryCondition(Choice): |
65 | """Change the victory condition. | 103 | """Change the victory condition. |
66 | On "the_end", the goal is to solve THE END at the top of the tower. | 104 | On "the_end", the goal is to solve THE END at the top of the tower. |
67 | On "the_master", the goal is to solve THE MASTER at the top of the tower, after getting the number of achievements specified in the Mastery Achievements option. | 105 | On "the_master", the goal is to solve THE MASTER at the top of the tower, after getting the number of achievements specified in the Mastery Achievements option. |
68 | On "level_2", the goal is to solve LEVEL 2 in the second room, after solving the number of panels specified in the Level 2 Requirement option.""" | 106 | On "level_2", the goal is to solve LEVEL 2 in the second room, after solving the number of panels specified in the Level 2 Requirement option. |
107 | On "pilgrimage", the goal is to solve PILGRIM in the Pilgrim Antechamber, typically after performing a Pilgrimage.""" | ||
69 | display_name = "Victory Condition" | 108 | display_name = "Victory Condition" |
70 | option_the_end = 0 | 109 | option_the_end = 0 |
71 | option_the_master = 1 | 110 | option_the_master = 1 |
72 | option_level_2 = 2 | 111 | option_level_2 = 2 |
112 | option_pilgrimage = 3 | ||
73 | 113 | ||
74 | 114 | ||
75 | class MasteryAchievements(Range): | 115 | class MasteryAchievements(Range): |
@@ -140,6 +180,11 @@ class LingoOptions(PerGameCommonOptions): | |||
140 | shuffle_colors: ShuffleColors | 180 | shuffle_colors: ShuffleColors |
141 | shuffle_panels: ShufflePanels | 181 | shuffle_panels: ShufflePanels |
142 | shuffle_paintings: ShufflePaintings | 182 | shuffle_paintings: ShufflePaintings |
183 | enable_pilgrimage: EnablePilgrimage | ||
184 | pilgrimage_allows_roof_access: PilgrimageAllowsRoofAccess | ||
185 | pilgrimage_allows_paintings: PilgrimageAllowsPaintings | ||
186 | sunwarp_access: SunwarpAccess | ||
187 | shuffle_sunwarps: ShuffleSunwarps | ||
143 | victory_condition: VictoryCondition | 188 | victory_condition: VictoryCondition |
144 | mastery_achievements: MasteryAchievements | 189 | mastery_achievements: MasteryAchievements |
145 | level_2_requirement: Level2Requirement | 190 | level_2_requirement: Level2Requirement |
diff --git a/player_logic.py b/player_logic.py index 966f5a1..96e9869 100644 --- a/player_logic.py +++ b/player_logic.py | |||
@@ -1,12 +1,13 @@ | |||
1 | from enum import Enum | 1 | from enum import Enum |
2 | from typing import Dict, List, NamedTuple, Optional, Set, Tuple, TYPE_CHECKING | 2 | from typing import Dict, List, NamedTuple, Optional, Set, Tuple, TYPE_CHECKING |
3 | 3 | ||
4 | from .datatypes import Door, RoomAndDoor, RoomAndPanel | 4 | from .datatypes import Door, DoorType, RoomAndDoor, RoomAndPanel |
5 | from .items import ALL_ITEM_TABLE, ItemData | 5 | from .items import ALL_ITEM_TABLE, ItemType |
6 | from .locations import ALL_LOCATION_TABLE, LocationClassification | 6 | from .locations import ALL_LOCATION_TABLE, LocationClassification |
7 | from .options import LocationChecks, ShuffleDoors, VictoryCondition | 7 | from .options import LocationChecks, ShuffleDoors, SunwarpAccess, VictoryCondition |
8 | from .static_logic import DOORS_BY_ROOM, PAINTINGS, PAINTING_ENTRANCES, PAINTING_EXITS, \ | 8 | from .static_logic import DOORS_BY_ROOM, PAINTINGS, PAINTING_ENTRANCES, PAINTING_EXITS, \ |
9 | PANELS_BY_ROOM, PROGRESSION_BY_ROOM, REQUIRED_PAINTING_ROOMS, REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS | 9 | PANELS_BY_ROOM, PROGRESSION_BY_ROOM, REQUIRED_PAINTING_ROOMS, REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS, \ |
10 | SUNWARP_ENTRANCES, SUNWARP_EXITS | ||
10 | 11 | ||
11 | if TYPE_CHECKING: | 12 | if TYPE_CHECKING: |
12 | from . import LingoWorld | 13 | from . import LingoWorld |
@@ -58,21 +59,6 @@ def should_split_progression(progression_name: str, world: "LingoWorld") -> Prog | |||
58 | return ProgressiveItemBehavior.PROGRESSIVE | 59 | return ProgressiveItemBehavior.PROGRESSIVE |
59 | 60 | ||
60 | 61 | ||
61 | def should_include_item(item: ItemData, world: "LingoWorld") -> bool: | ||
62 | if item.mode == "colors": | ||
63 | return world.options.shuffle_colors > 0 | ||
64 | elif item.mode == "doors": | ||
65 | return world.options.shuffle_doors != ShuffleDoors.option_none | ||
66 | elif item.mode == "complex door": | ||
67 | return world.options.shuffle_doors == ShuffleDoors.option_complex | ||
68 | elif item.mode == "door group": | ||
69 | return world.options.shuffle_doors == ShuffleDoors.option_simple | ||
70 | elif item.mode == "special": | ||
71 | return False | ||
72 | else: | ||
73 | return True | ||
74 | |||
75 | |||
76 | class LingoPlayerLogic: | 62 | class LingoPlayerLogic: |
77 | """ | 63 | """ |
78 | Defines logic after a player's options have been applied | 64 | Defines logic after a player's options have been applied |
@@ -99,6 +85,10 @@ class LingoPlayerLogic: | |||
99 | mastery_reqs: List[AccessRequirements] | 85 | mastery_reqs: List[AccessRequirements] |
100 | counting_panel_reqs: Dict[str, List[Tuple[AccessRequirements, int]]] | 86 | counting_panel_reqs: Dict[str, List[Tuple[AccessRequirements, int]]] |
101 | 87 | ||
88 | sunwarp_mapping: List[int] | ||
89 | sunwarp_entrances: List[str] | ||
90 | sunwarp_exits: List[str] | ||
91 | |||
102 | def add_location(self, room: str, name: str, code: Optional[int], panels: List[RoomAndPanel], world: "LingoWorld"): | 92 | def add_location(self, room: str, name: str, code: Optional[int], panels: List[RoomAndPanel], world: "LingoWorld"): |
103 | """ | 93 | """ |
104 | Creates a location. This function determines the access requirements for the location by combining and | 94 | Creates a location. This function determines the access requirements for the location by combining and |
@@ -132,6 +122,7 @@ class LingoPlayerLogic: | |||
132 | self.real_items.append(progressive_item_name) | 122 | self.real_items.append(progressive_item_name) |
133 | else: | 123 | else: |
134 | self.set_door_item(room_name, door_data.name, door_data.item_name) | 124 | self.set_door_item(room_name, door_data.name, door_data.item_name) |
125 | self.real_items.append(door_data.item_name) | ||
135 | 126 | ||
136 | def __init__(self, world: "LingoWorld"): | 127 | def __init__(self, world: "LingoWorld"): |
137 | self.item_by_door = {} | 128 | self.item_by_door = {} |
@@ -148,6 +139,7 @@ class LingoPlayerLogic: | |||
148 | self.door_reqs = {} | 139 | self.door_reqs = {} |
149 | self.mastery_reqs = [] | 140 | self.mastery_reqs = [] |
150 | self.counting_panel_reqs = {} | 141 | self.counting_panel_reqs = {} |
142 | self.sunwarp_mapping = [] | ||
151 | 143 | ||
152 | door_shuffle = world.options.shuffle_doors | 144 | door_shuffle = world.options.shuffle_doors |
153 | color_shuffle = world.options.shuffle_colors | 145 | color_shuffle = world.options.shuffle_colors |
@@ -161,15 +153,37 @@ class LingoPlayerLogic: | |||
161 | "be enough locations for all of the door items.") | 153 | "be enough locations for all of the door items.") |
162 | 154 | ||
163 | # Create door items, where needed. | 155 | # Create door items, where needed. |
164 | if door_shuffle != ShuffleDoors.option_none: | 156 | door_groups: Set[str] = set() |
165 | for room_name, room_data in DOORS_BY_ROOM.items(): | 157 | for room_name, room_data in DOORS_BY_ROOM.items(): |
166 | for door_name, door_data in room_data.items(): | 158 | for door_name, door_data in room_data.items(): |
167 | if door_data.skip_item is False and door_data.event is False: | 159 | if door_data.skip_item is False and door_data.event is False: |
160 | if door_data.type == DoorType.NORMAL and door_shuffle != ShuffleDoors.option_none: | ||
168 | if door_data.door_group is not None and door_shuffle == ShuffleDoors.option_simple: | 161 | if door_data.door_group is not None and door_shuffle == ShuffleDoors.option_simple: |
169 | # Grouped doors are handled differently if shuffle doors is on simple. | 162 | # Grouped doors are handled differently if shuffle doors is on simple. |
170 | self.set_door_item(room_name, door_name, door_data.door_group) | 163 | self.set_door_item(room_name, door_name, door_data.door_group) |
164 | door_groups.add(door_data.door_group) | ||
171 | else: | 165 | else: |
172 | self.handle_non_grouped_door(room_name, door_data, world) | 166 | self.handle_non_grouped_door(room_name, door_data, world) |
167 | elif door_data.type == DoorType.SUNWARP: | ||
168 | if world.options.sunwarp_access == SunwarpAccess.option_unlock: | ||
169 | self.set_door_item(room_name, door_name, "Sunwarps") | ||
170 | door_groups.add("Sunwarps") | ||
171 | elif world.options.sunwarp_access == SunwarpAccess.option_individual: | ||
172 | self.set_door_item(room_name, door_name, door_data.item_name) | ||
173 | self.real_items.append(door_data.item_name) | ||
174 | elif world.options.sunwarp_access == SunwarpAccess.option_progressive: | ||
175 | self.set_door_item(room_name, door_name, "Progressive Pilgrimage") | ||
176 | self.real_items.append("Progressive Pilgrimage") | ||
177 | elif door_data.type == DoorType.SUN_PAINTING: | ||
178 | if not world.options.enable_pilgrimage: | ||
179 | self.set_door_item(room_name, door_name, door_data.item_name) | ||
180 | self.real_items.append(door_data.item_name) | ||
181 | |||
182 | self.real_items += door_groups | ||
183 | |||
184 | # Create color items, if needed. | ||
185 | if color_shuffle: | ||
186 | self.real_items += [name for name, item in ALL_ITEM_TABLE.items() if item.type == ItemType.COLOR] | ||
173 | 187 | ||
174 | # Create events for each achievement panel, so that we can determine when THE MASTER is accessible. | 188 | # Create events for each achievement panel, so that we can determine when THE MASTER is accessible. |
175 | for room_name, room_data in PANELS_BY_ROOM.items(): | 189 | for room_name, room_data in PANELS_BY_ROOM.items(): |
@@ -206,6 +220,11 @@ class LingoPlayerLogic: | |||
206 | 220 | ||
207 | if world.options.level_2_requirement == 1: | 221 | if world.options.level_2_requirement == 1: |
208 | raise Exception("The Level 2 requirement must be at least 2 when LEVEL 2 is the victory condition.") | 222 | raise Exception("The Level 2 requirement must be at least 2 when LEVEL 2 is the victory condition.") |
223 | elif victory_condition == VictoryCondition.option_pilgrimage: | ||
224 | self.victory_condition = "Pilgrim Antechamber - PILGRIM" | ||
225 | self.add_location("Pilgrim Antechamber", "PILGRIM (Solved)", None, | ||
226 | [RoomAndPanel("Pilgrim Antechamber", "PILGRIM")], world) | ||
227 | self.event_loc_to_item["PILGRIM (Solved)"] = "Victory" | ||
209 | 228 | ||
210 | # Create groups of counting panel access requirements for the LEVEL 2 check. | 229 | # Create groups of counting panel access requirements for the LEVEL 2 check. |
211 | self.create_panel_hunt_events(world) | 230 | self.create_panel_hunt_events(world) |
@@ -225,28 +244,22 @@ class LingoPlayerLogic: | |||
225 | self.add_location(location_data.room, location_name, location_data.code, location_data.panels, world) | 244 | self.add_location(location_data.room, location_name, location_data.code, location_data.panels, world) |
226 | self.real_locations.append(location_name) | 245 | self.real_locations.append(location_name) |
227 | 246 | ||
228 | # Instantiate all real items. | 247 | if world.options.enable_pilgrimage and world.options.sunwarp_access == SunwarpAccess.option_disabled: |
229 | for name, item in ALL_ITEM_TABLE.items(): | 248 | raise Exception("Sunwarps cannot be disabled when pilgrimage is enabled.") |
230 | if should_include_item(item, world): | 249 | |
231 | self.real_items.append(name) | 250 | if world.options.shuffle_sunwarps: |
232 | 251 | if world.options.sunwarp_access == SunwarpAccess.option_disabled: | |
233 | # Calculate the requirements for the fake pilgrimage. | 252 | raise Exception("Sunwarps cannot be shuffled if they are disabled.") |
234 | fake_pilgrimage = [ | 253 | |
235 | ["Second Room", "Exit Door"], ["Crossroads", "Tower Entrance"], | 254 | self.sunwarp_mapping = list(range(0, 12)) |
236 | ["Orange Tower Fourth Floor", "Hot Crusts Door"], ["Outside The Initiated", "Shortcut to Hub Room"], | 255 | world.random.shuffle(self.sunwarp_mapping) |
237 | ["Orange Tower First Floor", "Shortcut to Hub Room"], ["Directional Gallery", "Shortcut to The Undeterred"], | 256 | |
238 | ["Orange Tower First Floor", "Salt Pepper Door"], ["Hub Room", "Crossroads Entrance"], | 257 | sunwarp_rooms = SUNWARP_ENTRANCES + SUNWARP_EXITS |
239 | ["Color Hunt", "Shortcut to The Steady"], ["The Bearer", "Entrance"], ["Art Gallery", "Exit"], | 258 | self.sunwarp_entrances = [sunwarp_rooms[i] for i in self.sunwarp_mapping[0:6]] |
240 | ["The Tenacious", "Shortcut to Hub Room"], ["Outside The Agreeable", "Tenacious Entrance"] | 259 | self.sunwarp_exits = [sunwarp_rooms[i] for i in self.sunwarp_mapping[6:12]] |
241 | ] | 260 | else: |
242 | pilgrimage_reqs = AccessRequirements() | 261 | self.sunwarp_entrances = SUNWARP_ENTRANCES |
243 | for door in fake_pilgrimage: | 262 | self.sunwarp_exits = SUNWARP_EXITS |
244 | door_object = DOORS_BY_ROOM[door[0]][door[1]] | ||
245 | if door_object.event or world.options.shuffle_doors == ShuffleDoors.option_none: | ||
246 | pilgrimage_reqs.merge(self.calculate_door_requirements(door[0], door[1], world)) | ||
247 | else: | ||
248 | pilgrimage_reqs.doors.add(RoomAndDoor(door[0], door[1])) | ||
249 | self.door_reqs.setdefault("Pilgrim Antechamber", {})["Pilgrimage"] = pilgrimage_reqs | ||
250 | 263 | ||
251 | # Create the paintings mapping, if painting shuffle is on. | 264 | # Create the paintings mapping, if painting shuffle is on. |
252 | if painting_shuffle: | 265 | if painting_shuffle: |
@@ -277,10 +290,11 @@ class LingoPlayerLogic: | |||
277 | # Starting Room - Exit Door gives access to OPEN and TRACE. | 290 | # Starting Room - Exit Door gives access to OPEN and TRACE. |
278 | good_item_options: List[str] = ["Starting Room - Back Right Door", "Second Room - Exit Door"] | 291 | good_item_options: List[str] = ["Starting Room - Back Right Door", "Second Room - Exit Door"] |
279 | 292 | ||
280 | if not color_shuffle: | 293 | if not color_shuffle and not world.options.enable_pilgrimage: |
281 | # HOT CRUST and THIS. | 294 | # HOT CRUST and THIS. |
282 | good_item_options.append("Pilgrim Room - Sun Painting") | 295 | good_item_options.append("Pilgrim Room - Sun Painting") |
283 | 296 | ||
297 | if not color_shuffle: | ||
284 | if door_shuffle == ShuffleDoors.option_simple: | 298 | if door_shuffle == ShuffleDoors.option_simple: |
285 | # WELCOME BACK, CLOCKWISE, and DRAWL + RUNS. | 299 | # WELCOME BACK, CLOCKWISE, and DRAWL + RUNS. |
286 | good_item_options.append("Welcome Back Doors") | 300 | good_item_options.append("Welcome Back Doors") |
diff --git a/regions.py b/regions.py index 5fddabd..4b357db 100644 --- a/regions.py +++ b/regions.py | |||
@@ -1,10 +1,11 @@ | |||
1 | from typing import Dict, Optional, TYPE_CHECKING | 1 | from typing import Dict, Optional, TYPE_CHECKING |
2 | 2 | ||
3 | from BaseClasses import Entrance, ItemClassification, Region | 3 | from BaseClasses import Entrance, ItemClassification, Region |
4 | from .datatypes import Room, RoomAndDoor | 4 | from .datatypes import EntranceType, Room, RoomAndDoor |
5 | from .items import LingoItem | 5 | from .items import LingoItem |
6 | from .locations import LingoLocation | 6 | from .locations import LingoLocation |
7 | from .rules import lingo_can_use_entrance, make_location_lambda | 7 | from .options import SunwarpAccess |
8 | from .rules import lingo_can_do_pilgrimage, lingo_can_use_entrance, make_location_lambda | ||
8 | from .static_logic import ALL_ROOMS, PAINTINGS | 9 | from .static_logic import ALL_ROOMS, PAINTINGS |
9 | 10 | ||
10 | if TYPE_CHECKING: | 11 | if 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 | ||
29 | def 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 | |||
28 | def connect_entrance(regions: Dict[str, Region], source_region: Region, target_region: Region, description: str, | 41 | def 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 | ||
43 | def connect_painting(regions: Dict[str, Region], warp_enter: str, warp_exit: str, world: "LingoWorld") -> None: | 71 | def 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 | ||
54 | def create_regions(world: "LingoWorld") -> None: | 83 | def 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() |
diff --git a/rules.py b/rules.py index 4e12938..9cc11fd 100644 --- a/rules.py +++ b/rules.py | |||
@@ -17,6 +17,10 @@ def lingo_can_use_entrance(state: CollectionState, room: str, door: RoomAndDoor, | |||
17 | return _lingo_can_open_door(state, effective_room, door.door, world) | 17 | return _lingo_can_open_door(state, effective_room, door.door, world) |
18 | 18 | ||
19 | 19 | ||
20 | def lingo_can_do_pilgrimage(state: CollectionState, world: "LingoWorld"): | ||
21 | return all(_lingo_can_open_door(state, "Sunwarps", f"{i} Sunwarp", world) for i in range(1, 7)) | ||
22 | |||
23 | |||
20 | def lingo_can_use_location(state: CollectionState, location: PlayerLocation, world: "LingoWorld"): | 24 | def lingo_can_use_location(state: CollectionState, location: PlayerLocation, world: "LingoWorld"): |
21 | return _lingo_can_satisfy_requirements(state, location.access, world) | 25 | return _lingo_can_satisfy_requirements(state, location.access, world) |
22 | 26 | ||
diff --git a/static_logic.py b/static_logic.py index 1da265d..c7ee001 100644 --- a/static_logic.py +++ b/static_logic.py | |||
@@ -1,10 +1,9 @@ | |||
1 | import os | 1 | import os |
2 | import pkgutil | 2 | import pkgutil |
3 | import pickle | ||
3 | from io import BytesIO | 4 | from io import BytesIO |
4 | from typing import Dict, List, Set | 5 | from typing import Dict, List, Set |
5 | 6 | ||
6 | import pickle | ||
7 | |||
8 | from .datatypes import Door, Painting, Panel, Progression, Room | 7 | from .datatypes import Door, Painting, Panel, Progression, Room |
9 | 8 | ||
10 | ALL_ROOMS: List[Room] = [] | 9 | ALL_ROOMS: List[Room] = [] |
@@ -21,6 +20,9 @@ PAINTING_EXITS: int = 0 | |||
21 | REQUIRED_PAINTING_ROOMS: List[str] = [] | 20 | REQUIRED_PAINTING_ROOMS: List[str] = [] |
22 | REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS: List[str] = [] | 21 | REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS: List[str] = [] |
23 | 22 | ||
23 | SUNWARP_ENTRANCES: List[str] = [] | ||
24 | SUNWARP_EXITS: List[str] = [] | ||
25 | |||
24 | SPECIAL_ITEM_IDS: Dict[str, int] = {} | 26 | SPECIAL_ITEM_IDS: Dict[str, int] = {} |
25 | PANEL_LOCATION_IDS: Dict[str, Dict[str, int]] = {} | 27 | PANEL_LOCATION_IDS: Dict[str, Dict[str, int]] = {} |
26 | DOOR_LOCATION_IDS: Dict[str, Dict[str, int]] = {} | 28 | DOOR_LOCATION_IDS: Dict[str, Dict[str, int]] = {} |
@@ -99,6 +101,8 @@ def load_static_data_from_file(): | |||
99 | PAINTING_EXITS = pickdata["PAINTING_EXITS"] | 101 | PAINTING_EXITS = pickdata["PAINTING_EXITS"] |
100 | REQUIRED_PAINTING_ROOMS.extend(pickdata["REQUIRED_PAINTING_ROOMS"]) | 102 | REQUIRED_PAINTING_ROOMS.extend(pickdata["REQUIRED_PAINTING_ROOMS"]) |
101 | REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS.extend(pickdata["REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS"]) | 103 | REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS.extend(pickdata["REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS"]) |
104 | SUNWARP_ENTRANCES.extend(pickdata["SUNWARP_ENTRANCES"]) | ||
105 | SUNWARP_EXITS.extend(pickdata["SUNWARP_EXITS"]) | ||
102 | SPECIAL_ITEM_IDS.update(pickdata["SPECIAL_ITEM_IDS"]) | 106 | SPECIAL_ITEM_IDS.update(pickdata["SPECIAL_ITEM_IDS"]) |
103 | PANEL_LOCATION_IDS.update(pickdata["PANEL_LOCATION_IDS"]) | 107 | PANEL_LOCATION_IDS.update(pickdata["PANEL_LOCATION_IDS"]) |
104 | DOOR_LOCATION_IDS.update(pickdata["DOOR_LOCATION_IDS"]) | 108 | DOOR_LOCATION_IDS.update(pickdata["DOOR_LOCATION_IDS"]) |
diff --git a/test/TestOptions.py b/test/TestOptions.py index 1769677..fce0743 100644 --- a/test/TestOptions.py +++ b/test/TestOptions.py | |||
@@ -29,3 +29,23 @@ class TestAllPanelHunt(LingoTestBase): | |||
29 | "level_2_requirement": "800", | 29 | "level_2_requirement": "800", |
30 | "early_color_hallways": "true" | 30 | "early_color_hallways": "true" |
31 | } | 31 | } |
32 | |||
33 | |||
34 | class TestShuffleSunwarps(LingoTestBase): | ||
35 | options = { | ||
36 | "shuffle_doors": "none", | ||
37 | "shuffle_colors": "false", | ||
38 | "victory_condition": "pilgrimage", | ||
39 | "shuffle_sunwarps": "true", | ||
40 | "sunwarp_access": "normal" | ||
41 | } | ||
42 | |||
43 | |||
44 | class TestShuffleSunwarpsAccess(LingoTestBase): | ||
45 | options = { | ||
46 | "shuffle_doors": "none", | ||
47 | "shuffle_colors": "false", | ||
48 | "victory_condition": "pilgrimage", | ||
49 | "shuffle_sunwarps": "true", | ||
50 | "sunwarp_access": "individual" | ||
51 | } \ No newline at end of file | ||
diff --git a/test/TestPilgrimage.py b/test/TestPilgrimage.py new file mode 100644 index 0000000..3cc9194 --- /dev/null +++ b/test/TestPilgrimage.py | |||
@@ -0,0 +1,114 @@ | |||
1 | from . import LingoTestBase | ||
2 | |||
3 | |||
4 | class TestDisabledPilgrimage(LingoTestBase): | ||
5 | options = { | ||
6 | "enable_pilgrimage": "false", | ||
7 | "shuffle_colors": "false" | ||
8 | } | ||
9 | |||
10 | def test_access(self): | ||
11 | self.assertFalse(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
12 | |||
13 | self.collect_by_name("Pilgrim Room - Sun Painting") | ||
14 | self.assertTrue(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
15 | |||
16 | |||
17 | class TestPilgrimageWithRoofAndPaintings(LingoTestBase): | ||
18 | options = { | ||
19 | "enable_pilgrimage": "true", | ||
20 | "shuffle_colors": "false", | ||
21 | "shuffle_doors": "complex", | ||
22 | "pilgrimage_allows_roof_access": "true", | ||
23 | "pilgrimage_allows_paintings": "true", | ||
24 | "early_color_hallways": "false" | ||
25 | } | ||
26 | |||
27 | def test_access(self): | ||
28 | doors = ["Second Room - Exit Door", "Crossroads - Roof Access", "Hub Room - Crossroads Entrance", | ||
29 | "Outside The Undeterred - Green Painting"] | ||
30 | |||
31 | for door in doors: | ||
32 | print(door) | ||
33 | self.assertFalse(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
34 | self.collect_by_name(door) | ||
35 | |||
36 | self.assertTrue(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
37 | |||
38 | |||
39 | class TestPilgrimageNoRoofYesPaintings(LingoTestBase): | ||
40 | options = { | ||
41 | "enable_pilgrimage": "true", | ||
42 | "shuffle_colors": "false", | ||
43 | "shuffle_doors": "complex", | ||
44 | "pilgrimage_allows_roof_access": "false", | ||
45 | "pilgrimage_allows_paintings": "true", | ||
46 | "early_color_hallways": "false" | ||
47 | } | ||
48 | |||
49 | def test_access(self): | ||
50 | doors = ["Second Room - Exit Door", "Crossroads - Roof Access", "Hub Room - Crossroads Entrance", | ||
51 | "Outside The Undeterred - Green Painting", "Crossroads - Tower Entrance", | ||
52 | "Orange Tower Fourth Floor - Hot Crusts Door", "Orange Tower First Floor - Shortcut to Hub Room", | ||
53 | "Starting Room - Street Painting"] | ||
54 | |||
55 | for door in doors: | ||
56 | print(door) | ||
57 | self.assertFalse(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
58 | self.collect_by_name(door) | ||
59 | |||
60 | self.assertTrue(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
61 | |||
62 | |||
63 | class TestPilgrimageNoRoofNoPaintings(LingoTestBase): | ||
64 | options = { | ||
65 | "enable_pilgrimage": "true", | ||
66 | "shuffle_colors": "false", | ||
67 | "shuffle_doors": "complex", | ||
68 | "pilgrimage_allows_roof_access": "false", | ||
69 | "pilgrimage_allows_paintings": "false", | ||
70 | "early_color_hallways": "false" | ||
71 | } | ||
72 | |||
73 | def test_access(self): | ||
74 | doors = ["Second Room - Exit Door", "Crossroads - Roof Access", "Hub Room - Crossroads Entrance", | ||
75 | "Outside The Undeterred - Green Painting", "Orange Tower First Floor - Shortcut to Hub Room", | ||
76 | "Starting Room - Street Painting", "Outside The Initiated - Shortcut to Hub Room", | ||
77 | "Directional Gallery - Shortcut to The Undeterred", "Orange Tower First Floor - Salt Pepper Door", | ||
78 | "Color Hunt - Shortcut to The Steady", "The Bearer - Entrance", | ||
79 | "Orange Tower Fifth Floor - Quadruple Intersection", "The Tenacious - Shortcut to Hub Room", | ||
80 | "Outside The Agreeable - Tenacious Entrance", "Crossroads - Tower Entrance", | ||
81 | "Orange Tower Fourth Floor - Hot Crusts Door"] | ||
82 | |||
83 | for door in doors: | ||
84 | print(door) | ||
85 | self.assertFalse(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
86 | self.collect_by_name(door) | ||
87 | |||
88 | self.assertTrue(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
89 | |||
90 | |||
91 | class TestPilgrimageYesRoofNoPaintings(LingoTestBase): | ||
92 | options = { | ||
93 | "enable_pilgrimage": "true", | ||
94 | "shuffle_colors": "false", | ||
95 | "shuffle_doors": "complex", | ||
96 | "pilgrimage_allows_roof_access": "true", | ||
97 | "pilgrimage_allows_paintings": "false", | ||
98 | "early_color_hallways": "false" | ||
99 | } | ||
100 | |||
101 | def test_access(self): | ||
102 | doors = ["Second Room - Exit Door", "Crossroads - Roof Access", "Hub Room - Crossroads Entrance", | ||
103 | "Outside The Undeterred - Green Painting", "Orange Tower First Floor - Shortcut to Hub Room", | ||
104 | "Starting Room - Street Painting", "Outside The Initiated - Shortcut to Hub Room", | ||
105 | "Directional Gallery - Shortcut to The Undeterred", "Orange Tower First Floor - Salt Pepper Door", | ||
106 | "Color Hunt - Shortcut to The Steady", "The Bearer - Entrance", | ||
107 | "Orange Tower Fifth Floor - Quadruple Intersection"] | ||
108 | |||
109 | for door in doors: | ||
110 | print(door) | ||
111 | self.assertFalse(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
112 | self.collect_by_name(door) | ||
113 | |||
114 | self.assertTrue(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
diff --git a/test/TestSunwarps.py b/test/TestSunwarps.py new file mode 100644 index 0000000..e8e913c --- /dev/null +++ b/test/TestSunwarps.py | |||
@@ -0,0 +1,213 @@ | |||
1 | from . import LingoTestBase | ||
2 | |||
3 | |||
4 | class TestVanillaDoorsNormalSunwarps(LingoTestBase): | ||
5 | options = { | ||
6 | "shuffle_doors": "none", | ||
7 | "shuffle_colors": "true", | ||
8 | "sunwarp_access": "normal" | ||
9 | } | ||
10 | |||
11 | def test_access(self): | ||
12 | self.assertTrue(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
13 | self.assertFalse(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
14 | |||
15 | self.collect_by_name("Yellow") | ||
16 | self.assertTrue(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
17 | self.assertTrue(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
18 | |||
19 | |||
20 | class TestSimpleDoorsNormalSunwarps(LingoTestBase): | ||
21 | options = { | ||
22 | "shuffle_doors": "simple", | ||
23 | "sunwarp_access": "normal" | ||
24 | } | ||
25 | |||
26 | def test_access(self): | ||
27 | self.assertFalse(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
28 | |||
29 | self.collect_by_name("Second Room - Exit Door") | ||
30 | self.assertTrue(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
31 | self.assertFalse(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
32 | |||
33 | self.collect_by_name(["Crossroads - Tower Entrances", "Orange Tower Fourth Floor - Hot Crusts Door"]) | ||
34 | self.assertTrue(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
35 | self.assertTrue(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
36 | |||
37 | |||
38 | class TestSimpleDoorsDisabledSunwarps(LingoTestBase): | ||
39 | options = { | ||
40 | "shuffle_doors": "simple", | ||
41 | "sunwarp_access": "disabled" | ||
42 | } | ||
43 | |||
44 | def test_access(self): | ||
45 | self.assertFalse(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
46 | |||
47 | self.collect_by_name("Second Room - Exit Door") | ||
48 | self.assertFalse(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
49 | self.assertFalse(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
50 | |||
51 | self.collect_by_name(["Hub Room - Crossroads Entrance", "Crossroads - Tower Entrancse", | ||
52 | "Orange Tower Fourth Floor - Hot Crusts Door"]) | ||
53 | self.assertFalse(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
54 | self.assertFalse(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
55 | |||
56 | |||
57 | class TestSimpleDoorsUnlockSunwarps(LingoTestBase): | ||
58 | options = { | ||
59 | "shuffle_doors": "simple", | ||
60 | "sunwarp_access": "unlock" | ||
61 | } | ||
62 | |||
63 | def test_access(self): | ||
64 | self.assertFalse(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
65 | |||
66 | self.collect_by_name("Second Room - Exit Door") | ||
67 | self.assertFalse(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
68 | |||
69 | self.collect_by_name(["Crossroads - Tower Entrances", "Orange Tower Fourth Floor - Hot Crusts Door"]) | ||
70 | self.assertFalse(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
71 | self.assertFalse(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
72 | |||
73 | self.collect_by_name("Sunwarps") | ||
74 | self.assertTrue(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
75 | self.assertTrue(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
76 | self.assertTrue(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
77 | |||
78 | |||
79 | class TestComplexDoorsNormalSunwarps(LingoTestBase): | ||
80 | options = { | ||
81 | "shuffle_doors": "complex", | ||
82 | "sunwarp_access": "normal" | ||
83 | } | ||
84 | |||
85 | def test_access(self): | ||
86 | self.assertFalse(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
87 | |||
88 | self.collect_by_name("Second Room - Exit Door") | ||
89 | self.assertTrue(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
90 | self.assertFalse(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
91 | |||
92 | self.collect_by_name(["Crossroads - Tower Entrance", "Orange Tower Fourth Floor - Hot Crusts Door"]) | ||
93 | self.assertTrue(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
94 | self.assertTrue(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
95 | |||
96 | |||
97 | class TestComplexDoorsDisabledSunwarps(LingoTestBase): | ||
98 | options = { | ||
99 | "shuffle_doors": "complex", | ||
100 | "sunwarp_access": "disabled" | ||
101 | } | ||
102 | |||
103 | def test_access(self): | ||
104 | self.assertFalse(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
105 | |||
106 | self.collect_by_name("Second Room - Exit Door") | ||
107 | self.assertFalse(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
108 | self.assertFalse(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
109 | |||
110 | self.collect_by_name(["Hub Room - Crossroads Entrance", "Crossroads - Tower Entrance", | ||
111 | "Orange Tower Fourth Floor - Hot Crusts Door"]) | ||
112 | self.assertFalse(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
113 | self.assertFalse(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
114 | |||
115 | |||
116 | class TestComplexDoorsIndividualSunwarps(LingoTestBase): | ||
117 | options = { | ||
118 | "shuffle_doors": "complex", | ||
119 | "sunwarp_access": "individual" | ||
120 | } | ||
121 | |||
122 | def test_access(self): | ||
123 | self.assertFalse(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
124 | |||
125 | self.collect_by_name("Second Room - Exit Door") | ||
126 | self.assertFalse(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
127 | |||
128 | self.collect_by_name("1 Sunwarp") | ||
129 | self.assertTrue(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
130 | |||
131 | self.collect_by_name(["Crossroads - Tower Entrance", "Orange Tower Fourth Floor - Hot Crusts Door"]) | ||
132 | self.assertFalse(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
133 | self.assertFalse(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
134 | |||
135 | self.collect_by_name("2 Sunwarp") | ||
136 | self.assertTrue(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
137 | self.assertFalse(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
138 | |||
139 | self.collect_by_name("3 Sunwarp") | ||
140 | self.assertTrue(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
141 | |||
142 | |||
143 | class TestComplexDoorsProgressiveSunwarps(LingoTestBase): | ||
144 | options = { | ||
145 | "shuffle_doors": "complex", | ||
146 | "sunwarp_access": "progressive" | ||
147 | } | ||
148 | |||
149 | def test_access(self): | ||
150 | self.assertFalse(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
151 | |||
152 | self.collect_by_name("Second Room - Exit Door") | ||
153 | self.assertFalse(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
154 | |||
155 | progressive_pilgrimage = self.get_items_by_name("Progressive Pilgrimage") | ||
156 | self.collect(progressive_pilgrimage[0]) | ||
157 | self.assertTrue(self.multiworld.state.can_reach("Crossroads", "Region", self.player)) | ||
158 | |||
159 | self.collect_by_name(["Crossroads - Tower Entrance", "Orange Tower Fourth Floor - Hot Crusts Door"]) | ||
160 | self.assertFalse(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
161 | self.assertFalse(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
162 | |||
163 | self.collect(progressive_pilgrimage[1]) | ||
164 | self.assertTrue(self.multiworld.state.can_reach("Orange Tower Third Floor", "Region", self.player)) | ||
165 | self.assertFalse(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
166 | |||
167 | self.collect(progressive_pilgrimage[2]) | ||
168 | self.assertTrue(self.multiworld.state.can_reach("Outside The Initiated", "Region", self.player)) | ||
169 | |||
170 | |||
171 | class TestUnlockSunwarpPilgrimage(LingoTestBase): | ||
172 | options = { | ||
173 | "sunwarp_access": "unlock", | ||
174 | "shuffle_colors": "false", | ||
175 | "enable_pilgrimage": "true" | ||
176 | } | ||
177 | |||
178 | def test_access(self): | ||
179 | self.assertFalse(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
180 | |||
181 | self.collect_by_name("Sunwarps") | ||
182 | |||
183 | self.assertTrue(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
184 | |||
185 | |||
186 | class TestIndividualSunwarpPilgrimage(LingoTestBase): | ||
187 | options = { | ||
188 | "sunwarp_access": "individual", | ||
189 | "shuffle_colors": "false", | ||
190 | "enable_pilgrimage": "true" | ||
191 | } | ||
192 | |||
193 | def test_access(self): | ||
194 | for i in range(1, 7): | ||
195 | self.assertFalse(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
196 | self.collect_by_name(f"{i} Sunwarp") | ||
197 | |||
198 | self.assertTrue(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
199 | |||
200 | |||
201 | class TestProgressiveSunwarpPilgrimage(LingoTestBase): | ||
202 | options = { | ||
203 | "sunwarp_access": "progressive", | ||
204 | "shuffle_colors": "false", | ||
205 | "enable_pilgrimage": "true" | ||
206 | } | ||
207 | |||
208 | def test_access(self): | ||
209 | for item in self.get_items_by_name("Progressive Pilgrimage"): | ||
210 | self.assertFalse(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
211 | self.collect(item) | ||
212 | |||
213 | self.assertTrue(self.can_reach_location("Pilgrim Antechamber - PILGRIM")) | ||
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 @@ | |||
1 | from typing import Dict, List, Set | 1 | from typing import Dict, List, Set, Optional |
2 | 2 | ||
3 | import os | 3 | import os |
4 | import sys | 4 | import sys |
@@ -6,7 +6,8 @@ import sys | |||
6 | sys.path.append(os.path.join("worlds", "lingo")) | 6 | sys.path.append(os.path.join("worlds", "lingo")) |
7 | sys.path.append(".") | 7 | sys.path.append(".") |
8 | sys.path.append("..") | 8 | sys.path.append("..") |
9 | from datatypes import Door, Painting, Panel, Progression, Room, RoomAndDoor, RoomAndPanel, RoomEntrance | 9 | from datatypes import Door, DoorType, EntranceType, Painting, Panel, Progression, Room, RoomAndDoor, RoomAndPanel,\ |
10 | RoomEntrance | ||
10 | 11 | ||
11 | import hashlib | 12 | import hashlib |
12 | import pickle | 13 | import pickle |
@@ -28,6 +29,9 @@ PAINTING_EXITS: int = 0 | |||
28 | REQUIRED_PAINTING_ROOMS: List[str] = [] | 29 | REQUIRED_PAINTING_ROOMS: List[str] = [] |
29 | REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS: List[str] = [] | 30 | REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS: List[str] = [] |
30 | 31 | ||
32 | SUNWARP_ENTRANCES: List[str] = ["", "", "", "", "", ""] | ||
33 | SUNWARP_EXITS: List[str] = ["", "", "", "", "", ""] | ||
34 | |||
31 | SPECIAL_ITEM_IDS: Dict[str, int] = {} | 35 | SPECIAL_ITEM_IDS: Dict[str, int] = {} |
32 | PANEL_LOCATION_IDS: Dict[str, Dict[str, int]] = {} | 36 | PANEL_LOCATION_IDS: Dict[str, Dict[str, int]] = {} |
33 | DOOR_LOCATION_IDS: Dict[str, Dict[str, int]] = {} | 37 | DOOR_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 | ||
99 | def process_entrance(source_room, doors, room_obj): | 103 | def 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 | |||
129 | def 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 | ||
136 | def process_panel(room_name, panel_name, panel_data): | 150 | def 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 | ||
395 | def 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 | |||
380 | def process_progression(room_name, progression_name, progression_doors): | 404 | def 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, |
diff --git a/utils/validate_config.rb b/utils/validate_config.rb index ae0ac61..831fee2 100644 --- a/utils/validate_config.rb +++ b/utils/validate_config.rb | |||
@@ -37,12 +37,14 @@ configured_panels = Set[] | |||
37 | mentioned_rooms = Set[] | 37 | mentioned_rooms = Set[] |
38 | mentioned_doors = Set[] | 38 | mentioned_doors = Set[] |
39 | mentioned_panels = Set[] | 39 | mentioned_panels = Set[] |
40 | mentioned_sunwarp_entrances = Set[] | ||
41 | mentioned_sunwarp_exits = Set[] | ||
40 | 42 | ||
41 | door_groups = {} | 43 | door_groups = {} |
42 | 44 | ||
43 | directives = Set["entrances", "panels", "doors", "paintings", "progression"] | 45 | directives = Set["entrances", "panels", "doors", "paintings", "sunwarps", "progression"] |
44 | panel_directives = Set["id", "required_room", "required_door", "required_panel", "colors", "check", "exclude_reduce", "tag", "link", "subtag", "achievement", "copy_to_sign", "non_counting", "hunt"] | 46 | panel_directives = Set["id", "required_room", "required_door", "required_panel", "colors", "check", "exclude_reduce", "tag", "link", "subtag", "achievement", "copy_to_sign", "non_counting", "hunt"] |
45 | door_directives = Set["id", "painting_id", "panels", "item_name", "item_group", "location_name", "skip_location", "skip_item", "door_group", "include_reduce", "junk_item", "event"] | 47 | door_directives = Set["id", "painting_id", "panels", "item_name", "item_group", "location_name", "skip_location", "skip_item", "door_group", "include_reduce", "event", "warp_id"] |
46 | painting_directives = Set["id", "enter_only", "exit_only", "orientation", "required_door", "required", "required_when_no_doors", "move", "req_blocked", "req_blocked_when_no_doors"] | 48 | painting_directives = Set["id", "enter_only", "exit_only", "orientation", "required_door", "required", "required_when_no_doors", "move", "req_blocked", "req_blocked_when_no_doors"] |
47 | 49 | ||
48 | non_counting = 0 | 50 | non_counting = 0 |
@@ -67,17 +69,17 @@ config.each do |room_name, room| | |||
67 | 69 | ||
68 | entrances = [] | 70 | entrances = [] |
69 | if entrance.kind_of? Hash | 71 | if entrance.kind_of? Hash |
70 | if entrance.keys() != ["painting"] then | 72 | entrances = [entrance] |
71 | entrances = [entrance] | ||
72 | end | ||
73 | elsif entrance.kind_of? Array | 73 | elsif entrance.kind_of? Array |
74 | entrances = entrance | 74 | entrances = entrance |
75 | end | 75 | end |
76 | 76 | ||
77 | entrances.each do |e| | 77 | entrances.each do |e| |
78 | entrance_room = e.include?("room") ? e["room"] : room_name | 78 | if e.include?("door") then |
79 | mentioned_rooms.add(entrance_room) | 79 | entrance_room = e.include?("room") ? e["room"] : room_name |
80 | mentioned_doors.add(entrance_room + " - " + e["door"]) | 80 | mentioned_rooms.add(entrance_room) |
81 | mentioned_doors.add(entrance_room + " - " + e["door"]) | ||
82 | end | ||
81 | end | 83 | end |
82 | end | 84 | end |
83 | 85 | ||
@@ -204,8 +206,8 @@ config.each do |room_name, room| | |||
204 | end | 206 | end |
205 | end | 207 | end |
206 | 208 | ||
207 | if not door.include?("id") and not door.include?("painting_id") and not door["skip_item"] and not door["event"] then | 209 | if not door.include?("id") and not door.include?("painting_id") and not door.include?("warp_id") and not door["skip_item"] and not door["event"] then |
208 | puts "#{room_name} - #{door_name} :::: Should be marked skip_item or event if there are no doors or paintings" | 210 | puts "#{room_name} - #{door_name} :::: Should be marked skip_item or event if there are no doors, paintings, or warps" |
209 | end | 211 | end |
210 | 212 | ||
211 | if door.include?("panels") | 213 | if door.include?("panels") |
@@ -292,6 +294,32 @@ config.each do |room_name, room| | |||
292 | end | 294 | end |
293 | end | 295 | end |
294 | 296 | ||
297 | (room["sunwarps"] || []).each do |sunwarp| | ||
298 | if sunwarp.include? "dots" and sunwarp.include? "direction" then | ||
299 | if sunwarp["dots"] < 1 or sunwarp["dots"] > 6 then | ||
300 | puts "#{room_name} :::: Contains a sunwarp with an invalid dots value" | ||
301 | end | ||
302 | |||
303 | if sunwarp["direction"] == "enter" then | ||
304 | if mentioned_sunwarp_entrances.include? sunwarp["dots"] then | ||
305 | puts "Multiple #{sunwarp["dots"]} sunwarp entrances were found" | ||
306 | else | ||
307 | mentioned_sunwarp_entrances.add(sunwarp["dots"]) | ||
308 | end | ||
309 | elsif sunwarp["direction"] == "exit" then | ||
310 | if mentioned_sunwarp_exits.include? sunwarp["dots"] then | ||
311 | puts "Multiple #{sunwarp["dots"]} sunwarp exits were found" | ||
312 | else | ||
313 | mentioned_sunwarp_exits.add(sunwarp["dots"]) | ||
314 | end | ||
315 | else | ||
316 | puts "#{room_name} :::: Contains a sunwarp with an invalid direction value" | ||
317 | end | ||
318 | else | ||
319 | puts "#{room_name} :::: Contains a sunwarp without a dots and direction" | ||
320 | end | ||
321 | end | ||
322 | |||
295 | (room["progression"] || {}).each do |progression_name, door_list| | 323 | (room["progression"] || {}).each do |progression_name, door_list| |
296 | door_list.each do |door| | 324 | door_list.each do |door| |
297 | if door.kind_of? Hash then | 325 | if door.kind_of? Hash then |