diff options
-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 |