diff options
Diffstat (limited to 'apworld/static_logic.py')
-rw-r--r-- | apworld/static_logic.py | 122 |
1 files changed, 116 insertions, 6 deletions
diff --git a/apworld/static_logic.py b/apworld/static_logic.py index 965ce3e..e59a47d 100644 --- a/apworld/static_logic.py +++ b/apworld/static_logic.py | |||
@@ -1,6 +1,8 @@ | |||
1 | from .generated import data_pb2 as data_pb2 | 1 | from .generated import data_pb2 as data_pb2 |
2 | from .items import SYMBOL_ITEMS, ANTI_COLLECTABLE_TRAPS | ||
2 | import pkgutil | 3 | import pkgutil |
3 | 4 | ||
5 | |||
4 | class Lingo2StaticLogic: | 6 | class Lingo2StaticLogic: |
5 | item_id_to_name: dict[int, str] | 7 | item_id_to_name: dict[int, str] |
6 | location_id_to_name: dict[int, str] | 8 | location_id_to_name: dict[int, str] |
@@ -8,9 +10,17 @@ class Lingo2StaticLogic: | |||
8 | item_name_to_id: dict[str, int] | 10 | item_name_to_id: dict[str, int] |
9 | location_name_to_id: dict[str, int] | 11 | location_name_to_id: dict[str, int] |
10 | 12 | ||
13 | item_name_groups: dict[str, list[str]] | ||
14 | location_name_groups: dict[str, list[str]] | ||
15 | |||
16 | letter_weights: dict[str, int] | ||
17 | |||
11 | def __init__(self): | 18 | def __init__(self): |
12 | self.item_id_to_name = {} | 19 | self.item_id_to_name = {} |
13 | self.location_id_to_name = {} | 20 | self.location_id_to_name = {} |
21 | self.item_name_groups = {} | ||
22 | self.location_name_groups = {} | ||
23 | self.letter_weights = {} | ||
14 | 24 | ||
15 | file = pkgutil.get_data(__name__, "generated/data.binpb") | 25 | file = pkgutil.get_data(__name__, "generated/data.binpb") |
16 | self.objects = data_pb2.AllObjects() | 26 | self.objects = data_pb2.AllObjects() |
@@ -18,38 +28,125 @@ class Lingo2StaticLogic: | |||
18 | 28 | ||
19 | for door in self.objects.doors: | 29 | for door in self.objects.doors: |
20 | if door.type in [data_pb2.DoorType.STANDARD, data_pb2.DoorType.LOCATION_ONLY, data_pb2.DoorType.GRAVESTONE]: | 30 | if door.type in [data_pb2.DoorType.STANDARD, data_pb2.DoorType.LOCATION_ONLY, data_pb2.DoorType.GRAVESTONE]: |
21 | location_name = f"{self.get_map_object_map_name(door)} - {door.name}" | 31 | location_name = self.get_door_location_name(door) |
22 | self.location_id_to_name[door.ap_id] = location_name | 32 | self.location_id_to_name[door.ap_id] = location_name |
23 | 33 | ||
24 | if door.type not in [data_pb2.DoorType.EVENT, data_pb2.DoorType.LOCATION_ONLY, data_pb2.DoorType.GRAVESTONE]: | 34 | if door.type not in [data_pb2.DoorType.EVENT, data_pb2.DoorType.LOCATION_ONLY, data_pb2.DoorType.GRAVESTONE]: |
25 | item_name = self.get_door_item_name(door.id) | 35 | item_name = self.get_door_item_name(door) |
26 | self.item_id_to_name[door.ap_id] = item_name | 36 | self.item_id_to_name[door.ap_id] = item_name |
27 | 37 | ||
28 | for letter in self.objects.letters: | 38 | for letter in self.objects.letters: |
29 | letter_name = f"{letter.key.upper()}{'2' if letter.level2 else '1'}" | 39 | letter_name = f"{letter.key.upper()}{'2' if letter.level2 else '1'}" |
30 | location_name = f"{self.get_room_object_map_name(letter)} - {letter_name}" | 40 | location_name = f"{self.get_room_object_map_name(letter)} - {letter_name}" |
31 | self.location_id_to_name[letter.ap_id] = location_name | 41 | self.location_id_to_name[letter.ap_id] = location_name |
42 | self.location_name_groups.setdefault("Letters", []).append(location_name) | ||
32 | 43 | ||
33 | if not letter.level2: | 44 | if not letter.level2: |
34 | self.item_id_to_name[letter.ap_id] = letter_name | 45 | self.item_id_to_name[letter.ap_id] = letter.key.upper() |
46 | self.item_name_groups.setdefault("Letters", []).append(letter.key.upper()) | ||
35 | 47 | ||
36 | for mastery in self.objects.masteries: | 48 | for mastery in self.objects.masteries: |
37 | location_name = f"{self.get_room_object_map_name(mastery)} - Mastery" | 49 | location_name = f"{self.get_room_object_map_name(mastery)} - Mastery" |
38 | self.location_id_to_name[mastery.ap_id] = location_name | 50 | self.location_id_to_name[mastery.ap_id] = location_name |
51 | self.location_name_groups.setdefault("Masteries", []).append(location_name) | ||
39 | 52 | ||
40 | for ending in self.objects.endings: | 53 | for ending in self.objects.endings: |
41 | location_name = f"{self.get_room_object_map_name(ending)} - {ending.name.title()} Ending" | 54 | location_name = f"{self.get_room_object_map_name(ending)} - {ending.name.title()} Ending" |
42 | self.location_id_to_name[ending.ap_id] = location_name | 55 | self.location_id_to_name[ending.ap_id] = location_name |
56 | self.location_name_groups.setdefault("Endings", []).append(location_name) | ||
57 | |||
58 | for progressive in self.objects.progressives: | ||
59 | self.item_id_to_name[progressive.ap_id] = progressive.name | ||
60 | |||
61 | for door_group in self.objects.door_groups: | ||
62 | self.item_id_to_name[door_group.ap_id] = door_group.name | ||
63 | |||
64 | for keyholder in self.objects.keyholders: | ||
65 | if keyholder.HasField("key"): | ||
66 | location_name = f"{self.get_room_object_location_prefix(keyholder)} - {keyholder.key.upper()} Keyholder" | ||
67 | self.location_id_to_name[keyholder.ap_id] = location_name | ||
68 | self.location_name_groups.setdefault("Keyholders", []).append(location_name) | ||
69 | |||
70 | self.item_id_to_name[self.objects.special_ids["A Job Well Done"]] = "A Job Well Done" | ||
71 | |||
72 | for symbol_name in SYMBOL_ITEMS.values(): | ||
73 | self.item_id_to_name[self.objects.special_ids[symbol_name]] = symbol_name | ||
43 | 74 | ||
44 | self.item_id_to_name[self.objects.special_ids["Nothing"]] = "Nothing" | 75 | for trap_name in ANTI_COLLECTABLE_TRAPS: |
76 | self.item_id_to_name[self.objects.special_ids[trap_name]] = trap_name | ||
45 | 77 | ||
46 | self.item_name_to_id = {name: ap_id for ap_id, name in self.item_id_to_name.items()} | 78 | self.item_name_to_id = {name: ap_id for ap_id, name in self.item_id_to_name.items()} |
47 | self.location_name_to_id = {name: ap_id for ap_id, name in self.location_id_to_name.items()} | 79 | self.location_name_to_id = {name: ap_id for ap_id, name in self.location_id_to_name.items()} |
48 | 80 | ||
49 | def get_door_item_name(self, door_id: int) -> str: | 81 | for panel in self.objects.panels: |
50 | door = self.objects.doors[door_id] | 82 | for letter in panel.answer.upper(): |
83 | self.letter_weights[letter] = self.letter_weights.get(letter, 0) + 1 | ||
84 | |||
85 | def get_door_item_name(self, door: data_pb2.Door) -> str: | ||
51 | return f"{self.get_map_object_map_name(door)} - {door.name}" | 86 | return f"{self.get_map_object_map_name(door)} - {door.name}" |
52 | 87 | ||
88 | def get_door_item_name_by_id(self, door_id: int) -> str: | ||
89 | door = self.objects.doors[door_id] | ||
90 | return self.get_door_item_name(door_id) | ||
91 | |||
92 | def get_door_location_name(self, door: data_pb2.Door) -> str: | ||
93 | map_part = self.get_room_object_location_prefix(door) | ||
94 | |||
95 | if door.HasField("location_name"): | ||
96 | return f"{map_part} - {door.location_name}" | ||
97 | |||
98 | generated_location_name = self.get_generated_door_location_name(door) | ||
99 | if generated_location_name is not None: | ||
100 | return generated_location_name | ||
101 | |||
102 | return f"{map_part} - {door.name}" | ||
103 | |||
104 | def get_generated_door_location_name(self, door: data_pb2.Door) -> str | None: | ||
105 | if door.type != data_pb2.DoorType.STANDARD: | ||
106 | return None | ||
107 | |||
108 | if len(door.keyholders) > 0 or len(door.endings) > 0 or door.HasField("complete_at"): | ||
109 | return None | ||
110 | |||
111 | if len(door.panels) > 4: | ||
112 | return None | ||
113 | |||
114 | map_areas = set() | ||
115 | for panel_id in door.panels: | ||
116 | panel = self.objects.panels[panel_id.panel] | ||
117 | panel_room = self.objects.rooms[panel.room_id] | ||
118 | # It's okay if panel_display_name is not present because then it's coalesced with other unnamed areas. | ||
119 | map_areas.add(panel_room.panel_display_name) | ||
120 | |||
121 | if len(map_areas) > 1: | ||
122 | return None | ||
123 | |||
124 | game_map = self.objects.maps[door.map_id] | ||
125 | map_area = map_areas.pop() | ||
126 | if map_area == "": | ||
127 | map_part = game_map.display_name | ||
128 | else: | ||
129 | map_part = f"{game_map.display_name} ({map_area})" | ||
130 | |||
131 | def get_panel_display_name(panel: data_pb2.ProxyIdentifier) -> str: | ||
132 | panel_data = self.objects.panels[panel.panel] | ||
133 | panel_name = panel_data.display_name if panel_data.HasField("display_name") else panel_data.name | ||
134 | |||
135 | if panel.HasField("answer"): | ||
136 | return f"{panel_name}/{panel.answer.upper()}" | ||
137 | else: | ||
138 | return panel_name | ||
139 | |||
140 | panel_names = [get_panel_display_name(panel_id) | ||
141 | for panel_id in door.panels] | ||
142 | panel_names.sort() | ||
143 | |||
144 | return map_part + " - " + ", ".join(panel_names) | ||
145 | |||
146 | def get_door_location_name_by_id(self, door_id: int) -> str: | ||
147 | door = self.objects.doors[door_id] | ||
148 | return self.get_door_location_name(door) | ||
149 | |||
53 | def get_room_region_name(self, room_id: int) -> str: | 150 | def get_room_region_name(self, room_id: int) -> str: |
54 | room = self.objects.rooms[room_id] | 151 | room = self.objects.rooms[room_id] |
55 | return f"{self.get_map_object_map_name(room)} - {room.name}" | 152 | return f"{self.get_map_object_map_name(room)} - {room.name}" |
@@ -59,3 +156,16 @@ class Lingo2StaticLogic: | |||
59 | 156 | ||
60 | def get_room_object_map_name(self, obj) -> str: | 157 | def get_room_object_map_name(self, obj) -> str: |
61 | return self.get_map_object_map_name(self.objects.rooms[obj.room_id]) | 158 | return self.get_map_object_map_name(self.objects.rooms[obj.room_id]) |
159 | |||
160 | def get_room_object_location_prefix(self, obj) -> str: | ||
161 | room = self.objects.rooms[obj.room_id] | ||
162 | game_map = self.objects.maps[room.map_id] | ||
163 | |||
164 | if room.HasField("panel_display_name"): | ||
165 | return f"{game_map.display_name} ({room.panel_display_name})" | ||
166 | else: | ||
167 | return game_map.display_name | ||
168 | |||
169 | def get_data_version(self) -> list[int]: | ||
170 | version = self.objects.version | ||
171 | return [version.major, version.minor, version.patch] | ||