summary refs log tree commit diff stats
diff options
context:
space:
mode:
-rw-r--r--__init__.py6
-rw-r--r--data/LL1.yaml587
-rw-r--r--data/generated.datbin130691 -> 135088 bytes
-rw-r--r--data/ids.yaml35
-rw-r--r--datatypes.py19
-rw-r--r--items.py39
-rw-r--r--locations.py2
-rw-r--r--options.py47
-rw-r--r--player_logic.py106
-rw-r--r--regions.py105
-rw-r--r--rules.py4
-rw-r--r--static_logic.py8
-rw-r--r--test/TestOptions.py20
-rw-r--r--test/TestPilgrimage.py114
-rw-r--r--test/TestSunwarps.py213
-rw-r--r--utils/pickle_static_data.py96
-rw-r--r--utils/validate_config.rb48
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
1455progression: 1473progression:
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 @@
1from enum import Enum, Flag, auto
1from typing import List, NamedTuple, Optional 2from 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
15class EntranceType(Flag):
16 NORMAL = auto()
17 PAINTING = auto()
18 SUNWARP = auto()
19 WARP = auto()
20 CROSSROADS_ROOF_ACCESS = auto()
21
22
14class RoomEntrance(NamedTuple): 23class 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
20class Room(NamedTuple): 29class Room(NamedTuple):
@@ -22,6 +31,12 @@ class Room(NamedTuple):
22 entrances: List[RoomEntrance] 31 entrances: List[RoomEntrance]
23 32
24 33
34class DoorType(Enum):
35 NORMAL = 1
36 SUNWARP = 2
37 SUN_PAINTING = 3
38
39
25class Door(NamedTuple): 40class 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 @@
1from typing import Dict, List, NamedTuple, Optional, TYPE_CHECKING 1from enum import Enum
2from typing import Dict, List, NamedTuple, Set
2 3
3from BaseClasses import Item, ItemClassification 4from BaseClasses import Item, ItemClassification
4from .static_logic import DOORS_BY_ROOM, PROGRESSION_BY_ROOM, PROGRESSIVE_ITEMS, get_door_group_item_id, \ 5from .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
9class ItemType(Enum):
10 NORMAL = 1
11 COLOR = 2
6 12
7 13
8class ItemData(NamedTuple): 14class 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
64class 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
70class 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
76class 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
82class 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
97class 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
64class VictoryCondition(Choice): 102class 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
75class MasteryAchievements(Range): 115class 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 @@
1from enum import Enum 1from enum import Enum
2from typing import Dict, List, NamedTuple, Optional, Set, Tuple, TYPE_CHECKING 2from typing import Dict, List, NamedTuple, Optional, Set, Tuple, TYPE_CHECKING
3 3
4from .datatypes import Door, RoomAndDoor, RoomAndPanel 4from .datatypes import Door, DoorType, RoomAndDoor, RoomAndPanel
5from .items import ALL_ITEM_TABLE, ItemData 5from .items import ALL_ITEM_TABLE, ItemType
6from .locations import ALL_LOCATION_TABLE, LocationClassification 6from .locations import ALL_LOCATION_TABLE, LocationClassification
7from .options import LocationChecks, ShuffleDoors, VictoryCondition 7from .options import LocationChecks, ShuffleDoors, SunwarpAccess, VictoryCondition
8from .static_logic import DOORS_BY_ROOM, PAINTINGS, PAINTING_ENTRANCES, PAINTING_EXITS, \ 8from .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
11if TYPE_CHECKING: 12if 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
61def 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
76class LingoPlayerLogic: 62class 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 @@
1from typing import Dict, Optional, TYPE_CHECKING 1from typing import Dict, Optional, TYPE_CHECKING
2 2
3from BaseClasses import Entrance, ItemClassification, Region 3from BaseClasses import Entrance, ItemClassification, Region
4from .datatypes import Room, RoomAndDoor 4from .datatypes import EntranceType, Room, RoomAndDoor
5from .items import LingoItem 5from .items import LingoItem
6from .locations import LingoLocation 6from .locations import LingoLocation
7from .rules import lingo_can_use_entrance, make_location_lambda 7from .options import SunwarpAccess
8from .rules import lingo_can_do_pilgrimage, lingo_can_use_entrance, make_location_lambda
8from .static_logic import ALL_ROOMS, PAINTINGS 9from .static_logic import ALL_ROOMS, PAINTINGS
9 10
10if TYPE_CHECKING: 11if 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
29def 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
28def connect_entrance(regions: Dict[str, Region], source_region: Region, target_region: Region, description: str, 41def 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
43def connect_painting(regions: Dict[str, Region], warp_enter: str, warp_exit: str, world: "LingoWorld") -> None: 71def 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
54def create_regions(world: "LingoWorld") -> None: 83def 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
20def 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
20def lingo_can_use_location(state: CollectionState, location: PlayerLocation, world: "LingoWorld"): 24def 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 @@
1import os 1import os
2import pkgutil 2import pkgutil
3import pickle
3from io import BytesIO 4from io import BytesIO
4from typing import Dict, List, Set 5from typing import Dict, List, Set
5 6
6import pickle
7
8from .datatypes import Door, Painting, Panel, Progression, Room 7from .datatypes import Door, Painting, Panel, Progression, Room
9 8
10ALL_ROOMS: List[Room] = [] 9ALL_ROOMS: List[Room] = []
@@ -21,6 +20,9 @@ PAINTING_EXITS: int = 0
21REQUIRED_PAINTING_ROOMS: List[str] = [] 20REQUIRED_PAINTING_ROOMS: List[str] = []
22REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS: List[str] = [] 21REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS: List[str] = []
23 22
23SUNWARP_ENTRANCES: List[str] = []
24SUNWARP_EXITS: List[str] = []
25
24SPECIAL_ITEM_IDS: Dict[str, int] = {} 26SPECIAL_ITEM_IDS: Dict[str, int] = {}
25PANEL_LOCATION_IDS: Dict[str, Dict[str, int]] = {} 27PANEL_LOCATION_IDS: Dict[str, Dict[str, int]] = {}
26DOOR_LOCATION_IDS: Dict[str, Dict[str, int]] = {} 28DOOR_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
34class 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
44class 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 @@
1from . import LingoTestBase
2
3
4class 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
17class 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
39class 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
63class 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
91class 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 @@
1from . import LingoTestBase
2
3
4class 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
20class 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
38class 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
57class 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
79class 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
97class 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
116class 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
143class 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
171class 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
186class 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
201class 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 @@
1from typing import Dict, List, Set 1from typing import Dict, List, Set, Optional
2 2
3import os 3import os
4import sys 4import sys
@@ -6,7 +6,8 @@ import sys
6sys.path.append(os.path.join("worlds", "lingo")) 6sys.path.append(os.path.join("worlds", "lingo"))
7sys.path.append(".") 7sys.path.append(".")
8sys.path.append("..") 8sys.path.append("..")
9from datatypes import Door, Painting, Panel, Progression, Room, RoomAndDoor, RoomAndPanel, RoomEntrance 9from datatypes import Door, DoorType, EntranceType, Painting, Panel, Progression, Room, RoomAndDoor, RoomAndPanel,\
10 RoomEntrance
10 11
11import hashlib 12import hashlib
12import pickle 13import pickle
@@ -28,6 +29,9 @@ PAINTING_EXITS: int = 0
28REQUIRED_PAINTING_ROOMS: List[str] = [] 29REQUIRED_PAINTING_ROOMS: List[str] = []
29REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS: List[str] = [] 30REQUIRED_PAINTING_WHEN_NO_DOORS_ROOMS: List[str] = []
30 31
32SUNWARP_ENTRANCES: List[str] = ["", "", "", "", "", ""]
33SUNWARP_EXITS: List[str] = ["", "", "", "", "", ""]
34
31SPECIAL_ITEM_IDS: Dict[str, int] = {} 35SPECIAL_ITEM_IDS: Dict[str, int] = {}
32PANEL_LOCATION_IDS: Dict[str, Dict[str, int]] = {} 36PANEL_LOCATION_IDS: Dict[str, Dict[str, int]] = {}
33DOOR_LOCATION_IDS: Dict[str, Dict[str, int]] = {} 37DOOR_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
99def process_entrance(source_room, doors, room_obj): 103def 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
129def 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
136def process_panel(room_name, panel_name, panel_data): 150def 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
395def 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
380def process_progression(room_name, progression_name, progression_doors): 404def 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[]
37mentioned_rooms = Set[] 37mentioned_rooms = Set[]
38mentioned_doors = Set[] 38mentioned_doors = Set[]
39mentioned_panels = Set[] 39mentioned_panels = Set[]
40mentioned_sunwarp_entrances = Set[]
41mentioned_sunwarp_exits = Set[]
40 42
41door_groups = {} 43door_groups = {}
42 44
43directives = Set["entrances", "panels", "doors", "paintings", "progression"] 45directives = Set["entrances", "panels", "doors", "paintings", "sunwarps", "progression"]
44panel_directives = Set["id", "required_room", "required_door", "required_panel", "colors", "check", "exclude_reduce", "tag", "link", "subtag", "achievement", "copy_to_sign", "non_counting", "hunt"] 46panel_directives = Set["id", "required_room", "required_door", "required_panel", "colors", "check", "exclude_reduce", "tag", "link", "subtag", "achievement", "copy_to_sign", "non_counting", "hunt"]
45door_directives = Set["id", "painting_id", "panels", "item_name", "item_group", "location_name", "skip_location", "skip_item", "door_group", "include_reduce", "junk_item", "event"] 47door_directives = Set["id", "painting_id", "panels", "item_name", "item_group", "location_name", "skip_location", "skip_item", "door_group", "include_reduce", "event", "warp_id"]
46painting_directives = Set["id", "enter_only", "exit_only", "orientation", "required_door", "required", "required_when_no_doors", "move", "req_blocked", "req_blocked_when_no_doors"] 48painting_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
48non_counting = 0 50non_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