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