mirror of
				https://github.com/MarioSpore/Grinch-AP.git
				synced 2025-10-21 20:21:32 -06:00 
			
		
		
		
	Compare commits
	
		
			96 Commits
		
	
	
		
			v1.0.0
			...
			5e71874446
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 5e71874446 | ||
|   | 1870dd24ba | ||
|   | f70b6c4c9c | ||
|   | 79d4d5b10b | ||
|   | 7fea34adc3 | ||
|   | a3f9e6cbc9 | ||
|   | bccc83f864 | ||
|   | 6409721841 | ||
|   | d3a7b014bd | ||
|   | 3ec8631203 | ||
|   | 2081912a39 | ||
|   | 67bbde2556 | ||
|   | 0503c2ead3 | ||
|   | 7a642cc1a9 | ||
|   | eb8d44e975 | ||
|   | 1f35f5fa93 | ||
|   | 6bc819f4bc | ||
|   | bb0c5f5b9a | ||
|   | 0e397c7079 | ||
|   | 2572a25089 | ||
|   | dcdf168618 | ||
|   | 8a8136a267 | ||
|   | 5f158497f9 | ||
|   | bc74e67e07 | ||
|   | dbc592dad0 | ||
|   | afe1345e34 | ||
|   | 410df2a948 | ||
|   | 7c6eada7b2 | ||
|   | 8e1217d1a5 | ||
|   | 5615277705 | ||
|   | 2de0f9d766 | ||
|   | 4da88cf794 | ||
|   | e2cc1b5de7 | ||
|   | 103a6f79d1 | ||
|   | 17861c1050 | ||
|   | a63c33a711 | ||
|   | c84ef117c8 | ||
|   | 834096f282 | ||
|   | 0a31d96ee4 | ||
|   | 3edb733dcb | ||
|   | 9c01cc31e0 | ||
|   | ea8262855e | ||
|   | 98ea11887e | ||
|   | 18aefcd3f2 | ||
|   | 3a13332533 | ||
|   | c279ef7bc6 | ||
|   | 55ef0cc8c9 | ||
|   | babc4f441c | ||
|   | 92d932da55 | ||
|   | b1b65a3adf | ||
|   | d6f5e87ccf | ||
|   | 079239ea70 | ||
|   | 93bac29e8c | ||
|   | c9fea29d92 | ||
|   | e17895902e | ||
|   | 1596550111 | ||
|   | c888e17845 | ||
|   | 3dee611b51 | ||
|   | d6c7a04316 | ||
|   | 900c8a519a | ||
|   | 43acc9f003 | ||
|   | 96eb8fcd9a | ||
|   | d4bd682ac9 | ||
|   | 61d4783f61 | ||
|   | 00fff466ff | ||
|   | d8483bef6e | ||
|   | 56a198fcfd | ||
|   | 4e362dc722 | ||
|   | cfcfc9ecfd | ||
|   | a3a415adfd | ||
|   | 14c95aa85b | ||
|   | 8d941dad6f | ||
|   | 8628f6637a | ||
|   | 17b7914c35 | ||
|   | b8dfd5ce4c | ||
|   | b3749b7fe3 | ||
|   | 3aaf625282 | ||
|   | 9df2360b8b | ||
|   | d61ac9a135 | ||
|   | c8fc56d7c4 | ||
|   | 51aad167cc | ||
|   | e2def66522 | ||
|   | 73e9d9d577 | ||
|   | a5d7ff65c1 | ||
|   | 05bf60abf7 | ||
|   | 7f627e2c07 | ||
|   | 19e0fe1286 | ||
|   | b390974019 | ||
|   | 9da65fab09 | ||
|   | 02d2eab5a4 | ||
|   | 985c8b681b | ||
|   | cf5a4012c0 | ||
|   | c59e75ef7b | ||
|   | 2dbe344348 | ||
|   | 90ba4fbda7 | ||
|   | 8ff2fb91d4 | 
| @@ -1,7 +1,10 @@ | |||||||
| from typing import TYPE_CHECKING | import time | ||||||
|  | from typing import TYPE_CHECKING, Sequence | ||||||
| import asyncio | import asyncio | ||||||
| import NetUtils | import NetUtils | ||||||
| import copy | import copy | ||||||
|  | import uuid | ||||||
|  | import Utils | ||||||
| from .Locations import grinch_locations, GrinchLocation | from .Locations import grinch_locations, GrinchLocation | ||||||
| from .Items import ALL_ITEMS_TABLE, MISSION_ITEMS_TABLE, GADGETS_TABLE, KEYS_TABLE, GrinchItemData #, SLEIGH_PARTS_TABLE | from .Items import ALL_ITEMS_TABLE, MISSION_ITEMS_TABLE, GADGETS_TABLE, KEYS_TABLE, GrinchItemData #, SLEIGH_PARTS_TABLE | ||||||
| import worlds._bizhawk as bizhawk | import worlds._bizhawk as bizhawk | ||||||
| @@ -23,20 +26,29 @@ MAX_DEMO_MODE_CHECK = 30 | |||||||
| # List of Menu Map IDs | # List of Menu Map IDs | ||||||
| MENU_MAP_IDS: list[int] = [0x00, 0x02, 0x35, 0x36, 0x37] | MENU_MAP_IDS: list[int] = [0x00, 0x02, 0x35, 0x36, 0x37] | ||||||
|  |  | ||||||
|  | MAX_EGGS: int = 200 | ||||||
|  | EGG_COUNT_ADDR: int = 0x010058 | ||||||
|  | EGG_ADDR_BYTESIZE: int = 2 | ||||||
|  |  | ||||||
| class GrinchClient(BizHawkClient): | class GrinchClient(BizHawkClient): | ||||||
|     game = "The Grinch" |     game = "The Grinch" | ||||||
|     system = "PSX" |     system = "PSX" | ||||||
|     patch_suffix = ".apgrinch" |     patch_suffix = ".apgrinch" | ||||||
|     items_handling = 0b111 |     items_handling = 0b111 | ||||||
|     demo_mode_buffer = 0 |     demo_mode_buffer: int = 0 | ||||||
|     last_map_location = -1 |     last_map_location: int = -1 | ||||||
|     ingame_log = False |     ingame_log: bool = False | ||||||
|  |     previous_egg_count: int = 0 | ||||||
|  |     send_ring_link: bool = False | ||||||
|  |     unique_client_id: int = 0 | ||||||
|  |     ring_link_enabled = False | ||||||
|  |  | ||||||
|     def __init__(self): |     def __init__(self): | ||||||
|         super().__init__() |         super().__init__() | ||||||
|         self.last_received_index = 0 |         self.last_received_index = 0 | ||||||
|         self.loading_bios_msg = False |         self.loading_bios_msg = False | ||||||
|         self.loc_unlimited_eggs = False |         self.loc_unlimited_eggs = False | ||||||
|  |         self.unique_client_id = 0 | ||||||
|  |  | ||||||
|     async def validate_rom(self, ctx: "BizHawkClientContext") -> bool: |     async def validate_rom(self, ctx: "BizHawkClientContext") -> bool: | ||||||
|         from CommonClient import logger |         from CommonClient import logger | ||||||
| @@ -59,13 +71,15 @@ class GrinchClient(BizHawkClient): | |||||||
|  |  | ||||||
|                 logger.error("Invalid rom detected. You are not playing Grinch USA Version.") |                 logger.error("Invalid rom detected. You are not playing Grinch USA Version.") | ||||||
|                 raise Exception("Invalid rom detected. You are not playing Grinch USA Version.") |                 raise Exception("Invalid rom detected. You are not playing Grinch USA Version.") | ||||||
|  |  | ||||||
|  |             ctx.command_processor.commands["ringlink"] = _cmd_ringlink | ||||||
|         except Exception: |         except Exception: | ||||||
|             return False |             return False | ||||||
|  |  | ||||||
|         ctx.game = self.game |         ctx.game = self.game | ||||||
|         ctx.items_handling = self.items_handling |         ctx.items_handling = self.items_handling | ||||||
|         ctx.want_slot_data = True |         ctx.want_slot_data = True | ||||||
|         ctx.watcher_timeout = 0.25 |         ctx.watcher_timeout = 0.125 | ||||||
|         self.loading_bios_msg = False |         self.loading_bios_msg = False | ||||||
|  |  | ||||||
|         return True |         return True | ||||||
| @@ -76,22 +90,46 @@ class GrinchClient(BizHawkClient): | |||||||
|         match cmd: |         match cmd: | ||||||
|             case "Connected":  # On Connect |             case "Connected":  # On Connect | ||||||
|                 self.loc_unlimited_eggs = bool(ctx.slot_data["give_unlimited_eggs"]) |                 self.loc_unlimited_eggs = bool(ctx.slot_data["give_unlimited_eggs"]) | ||||||
|  |                 self.unique_client_id = self._get_uuid() | ||||||
|                 logger.info("You are now connected to the client. "+ |                 logger.info("You are now connected to the client. "+ | ||||||
|                     "There may be a slight delay to check you are not in demo mode before locations start to send.") |                     "There may be a slight delay to check you are not in demo mode before locations start to send.") | ||||||
|  |  | ||||||
|  |                 self.ring_link_enabled = bool(ctx.slot_data["ring_link"]) | ||||||
|  |  | ||||||
|  |                 tags = copy.deepcopy(ctx.tags) | ||||||
|  |                 if self.ring_link_enabled: | ||||||
|  |                     ctx.tags.add("RingLink") | ||||||
|  |                 else: | ||||||
|  |                     ctx.tags -= { "RingLink" } | ||||||
|  |  | ||||||
|  |                 if tags != ctx.tags: | ||||||
|  |                     Utils.async_start(ctx.send_msgs([{"cmd": "ConnectUpdate", "tags": ctx.tags}]), "Update RingLink Tags") | ||||||
|  |  | ||||||
|  |             case "Bounced": | ||||||
|  |                 if "tags" not in args: | ||||||
|  |                     return | ||||||
|  |  | ||||||
|  |                 if "RingLink" in ctx.tags and "RingLink" in args["tags"] and args["data"]["source"] != self.unique_client_id: | ||||||
|  |                     Utils.async_start(self.ring_link_input(args["data"]["amount"], ctx), "SyncEggs") | ||||||
|  |  | ||||||
|     async def set_auth(self, ctx: "BizHawkClientContext") -> None: |     async def set_auth(self, ctx: "BizHawkClientContext") -> None: | ||||||
|         await ctx.get_username() |         await ctx.get_username() | ||||||
|  |  | ||||||
|     async def game_watcher(self, ctx: "BizHawkClientContext") -> None: |     async def game_watcher(self, ctx: "BizHawkClientContext") -> None: | ||||||
|         from CommonClient import  logger |         from CommonClient import logger | ||||||
|         #If the player is not connected to an AP Server, or their connection was disconnected. |         #If the player is not connected to an AP Server, or their connection was disconnected. | ||||||
|         if ctx.server is None or ctx.server.socket.closed or ctx.slot_data is None: |         if not ctx.slot: | ||||||
|             return |             return | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             if not await self.ingame_checker(ctx): |             if not await self.ingame_checker(ctx): | ||||||
|                 return |                 return | ||||||
|  |  | ||||||
|  |             if not any(task.get_name() == "Grinch EggLink" for task in asyncio.all_tasks()): | ||||||
|  |                 print("EggLink") | ||||||
|  |                 self.send_ring_link = True | ||||||
|  |                 Utils.async_start(self.ring_link_output(ctx), name="Grinch EggLink") | ||||||
|  |  | ||||||
|             await self.location_checker(ctx) |             await self.location_checker(ctx) | ||||||
|             await self.receiving_items_handler(ctx) |             await self.receiving_items_handler(ctx) | ||||||
|             await self.goal_checker(ctx) |             await self.goal_checker(ctx) | ||||||
| @@ -102,14 +140,32 @@ class GrinchClient(BizHawkClient): | |||||||
|             # The connector didn't respond. Exit handler and return to main loop to reconnect |             # The connector didn't respond. Exit handler and return to main loop to reconnect | ||||||
|             logger.error("Failure to connect / authenticate the grinch. Error details: " + str(ex)) |             logger.error("Failure to connect / authenticate the grinch. Error details: " + str(ex)) | ||||||
|             pass |             pass | ||||||
|  |         except Exception as genericEx: | ||||||
|  |             # For all other errors, catch this and let the client gracefully disconnect | ||||||
|  |             logger.error("Unknown error occurred while playing the grinch. Error details: " + str(genericEx)) | ||||||
|  |             await ctx.disconnect(False) | ||||||
|  |             pass | ||||||
|  |  | ||||||
|     async def location_checker(self, ctx: "BizHawkClientContext"): |     async def location_checker(self, ctx: "BizHawkClientContext"): | ||||||
|         from CommonClient import logger |         from CommonClient import logger | ||||||
|         # Update the AP Server to know what locations are not checked yet. |         # Update the AP Server to know what locations are not checked yet. | ||||||
|         local_locations_checked: list[int] = [] |         local_locations_checked: list[int] = [] | ||||||
|  |         addr_list_to_read: list[tuple[int, int, str]] = [] | ||||||
|         local_ap_locations: set[int] = copy.deepcopy(ctx.missing_locations) |         local_ap_locations: set[int] = copy.deepcopy(ctx.missing_locations) | ||||||
|  |  | ||||||
|  |         # Loop through the first time of everything left to create the list of RAM addresses to read / monitor. | ||||||
|  |         for missing_location in local_ap_locations: | ||||||
|  |             grinch_loc_name = ctx.location_names.lookup_in_game(missing_location) | ||||||
|  |             grinch_loc_ram_data = grinch_locations[grinch_loc_name] | ||||||
|  |             missing_addr_list: list[tuple[int, int, str]] = [(read_addr.ram_address, read_addr.bit_size, "MainRAM") for | ||||||
|  |                                                              read_addr in grinch_loc_ram_data.update_ram_addr] | ||||||
|  |             addr_list_to_read = [*addr_list_to_read, *missing_addr_list] | ||||||
|  |  | ||||||
|  |         returned_bytes: list[bytes] = await bizhawk.read(ctx.bizhawk_ctx, addr_list_to_read) | ||||||
|  |  | ||||||
|  |         # Now loop through everything again and this time get the byte value from the above read, convert to int, | ||||||
|  |         # and check to see if that ram address has our expected value. | ||||||
|         for missing_location in local_ap_locations: |         for missing_location in local_ap_locations: | ||||||
|             # local_location = ctx.location_names.lookup_in_game(missing_location) |  | ||||||
|             # Missing location is the AP ID & we need to convert it back to a location name within our game. |             # Missing location is the AP ID & we need to convert it back to a location name within our game. | ||||||
|             # Using the location name, we can then get the Grinch ram data from there. |             # Using the location name, we can then get the Grinch ram data from there. | ||||||
|             grinch_loc_name = ctx.location_names.lookup_in_game(missing_location) |             grinch_loc_name = ctx.location_names.lookup_in_game(missing_location) | ||||||
| @@ -120,15 +176,15 @@ class GrinchClient(BizHawkClient): | |||||||
|             ram_checked_list: list[bool] = [] |             ram_checked_list: list[bool] = [] | ||||||
|             for addr_to_update in grinch_loc_ram_data.update_ram_addr: |             for addr_to_update in grinch_loc_ram_data.update_ram_addr: | ||||||
|                 is_binary = True if not addr_to_update.binary_bit_pos is None else False |                 is_binary = True if not addr_to_update.binary_bit_pos is None else False | ||||||
|                 current_ram_address_value = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( |                 orig_index: int = addr_list_to_read.index((addr_to_update.ram_address, addr_to_update.bit_size, "MainRAM")) | ||||||
|                     addr_to_update.ram_address, addr_to_update.bit_size, "MainRAM")]))[0], "little") |                 value_read_from_bizhawk: int = int.from_bytes(returned_bytes[orig_index], "little") | ||||||
|                 if is_binary: |                 if is_binary: | ||||||
|                     ram_checked_list.append((current_ram_address_value & (1 << addr_to_update.binary_bit_pos)) > 0) |                     ram_checked_list.append((value_read_from_bizhawk & (1 << addr_to_update.binary_bit_pos)) > 0) | ||||||
|                 else: |                 else: | ||||||
|                     expected_int_value = addr_to_update.value |                     expected_int_value = addr_to_update.value | ||||||
|                     ram_checked_list.append(expected_int_value == current_ram_address_value) |                     ram_checked_list.append(expected_int_value == value_read_from_bizhawk) | ||||||
|                 if all(ram_checked_list): |             if all(ram_checked_list): | ||||||
|                     local_locations_checked.append(GrinchLocation.get_apid(grinch_loc_ram_data.id)) |                 local_locations_checked.append(GrinchLocation.get_apid(grinch_loc_ram_data.id)) | ||||||
|  |  | ||||||
|         # Update the AP server with the locally checked list of locations (In other words, locations I found in Grinch) |         # Update the AP server with the locally checked list of locations (In other words, locations I found in Grinch) | ||||||
|         locations_sent_to_ap: set[int] = await ctx.check_locations(local_locations_checked) |         locations_sent_to_ap: set[int] = await ctx.check_locations(local_locations_checked) | ||||||
| @@ -145,9 +201,9 @@ class GrinchClient(BizHawkClient): | |||||||
|             RECV_ITEM_ADDR, RECV_ITEM_BITSIZE, "MainRAM")]))[0], "little") |             RECV_ITEM_ADDR, RECV_ITEM_BITSIZE, "MainRAM")]))[0], "little") | ||||||
|         if len(ctx.items_received) == self.last_received_index: |         if len(ctx.items_received) == self.last_received_index: | ||||||
|             return |             return | ||||||
|  |  | ||||||
|         # Ensures we only get the new items that we want to give the player |         # Ensures we only get the new items that we want to give the player | ||||||
|         new_items_only = ctx.items_received[self.last_received_index:] |         new_items_only = ctx.items_received[self.last_received_index:] | ||||||
|  |         ram_addr_dict: dict[int, list[int]] = {} | ||||||
|  |  | ||||||
|         for item_received in new_items_only: |         for item_received in new_items_only: | ||||||
|             local_item = ctx.item_names.lookup_in_game(item_received.item) |             local_item = ctx.item_names.lookup_in_game(item_received.item) | ||||||
| @@ -155,8 +211,11 @@ class GrinchClient(BizHawkClient): | |||||||
|  |  | ||||||
|             for addr_to_update in grinch_item_ram_data.update_ram_addr: |             for addr_to_update in grinch_item_ram_data.update_ram_addr: | ||||||
|                 is_binary = True if not addr_to_update.binary_bit_pos is None else False |                 is_binary = True if not addr_to_update.binary_bit_pos is None else False | ||||||
|                 current_ram_address_value = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( |                 if addr_to_update.ram_address in ram_addr_dict.keys(): | ||||||
|                     addr_to_update.ram_address, addr_to_update.bit_size, "MainRAM")]))[0], "little") |                     current_ram_address_value = ram_addr_dict[addr_to_update.ram_address][0] | ||||||
|  |                 else: | ||||||
|  |                     current_ram_address_value = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( | ||||||
|  |                         addr_to_update.ram_address, addr_to_update.bit_size, "MainRAM")]))[0], "little") | ||||||
|                 if is_binary: |                 if is_binary: | ||||||
|                     current_ram_address_value = (current_ram_address_value | (1 << addr_to_update.binary_bit_pos)) |                     current_ram_address_value = (current_ram_address_value | (1 << addr_to_update.binary_bit_pos)) | ||||||
|                 elif addr_to_update.update_existing_value: |                 elif addr_to_update.update_existing_value: | ||||||
| @@ -167,20 +226,22 @@ class GrinchClient(BizHawkClient): | |||||||
|                     current_ram_address_value = addr_to_update.value |                     current_ram_address_value = addr_to_update.value | ||||||
|  |  | ||||||
|                 # Write the updated value back into RAM |                 # Write the updated value back into RAM | ||||||
|                 await self.update_and_validate_address(ctx, addr_to_update.ram_address, current_ram_address_value, addr_to_update.bit_size) |                 ram_addr_dict[addr_to_update.ram_address] = [current_ram_address_value, addr_to_update.bit_size] | ||||||
|                 # await bizhawk.write(ctx.bizhawk_ctx, [(addr_to_update.ram_address, |  | ||||||
|                 #     current_ram_address_value.to_bytes(addr_to_update.bit_size, "little"), "MainRAM")]) |  | ||||||
|  |  | ||||||
|             self.last_received_index += 1 |             self.last_received_index += 1 | ||||||
|             await self.update_and_validate_address(ctx, RECV_ITEM_ADDR, self.last_received_index, RECV_ITEM_BITSIZE) |  | ||||||
|  |         # Update the latest received item index to ram as well. | ||||||
|  |         ram_addr_dict[RECV_ITEM_ADDR] = [self.last_received_index, RECV_ITEM_BITSIZE] | ||||||
|  |         await bizhawk.write(ctx.bizhawk_ctx, self.convert_dict_to_ram_list(ram_addr_dict)) | ||||||
|  |  | ||||||
|     async def goal_checker(self, ctx: "BizHawkClientContext"): |     async def goal_checker(self, ctx: "BizHawkClientContext"): | ||||||
|         if not ctx.finished_game: |         if not ctx.finished_game: | ||||||
|             goal_loc = grinch_locations["Neutralizing Santa"] |             goal_loc = grinch_locations["MC - Sleigh Ride - Neutralizing Santa"] | ||||||
|             goal_ram_address = goal_loc.update_ram_addr[0] |             goal_ram_address = goal_loc.update_ram_addr[0] | ||||||
|             current_ram_address_value = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( |             current_ram_address_value = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( | ||||||
|                 goal_ram_address.ram_address, goal_ram_address.bit_size, "MainRAM")]))[0], "little") |                 goal_ram_address.ram_address, goal_ram_address.bit_size, "MainRAM")]))[0], "little") | ||||||
|             if (current_ram_address_value & (1 << goal_ram_address.binary_bit_pos)) > 0: |             # if (current_ram_address_value & (1 << goal_ram_address.binary_bit_pos)) > 0: | ||||||
|  |             if current_ram_address_value == goal_ram_address.value: | ||||||
|                 ctx.finished_game = True |                 ctx.finished_game = True | ||||||
|                 await ctx.send_msgs([{ |                 await ctx.send_msgs([{ | ||||||
|                     "cmd": "StatusUpdate", |                     "cmd": "StatusUpdate", | ||||||
| @@ -189,14 +250,16 @@ class GrinchClient(BizHawkClient): | |||||||
|  |  | ||||||
|     # This function's entire purpose is to take away items we physically received ingame, but have not received from AP |     # This function's entire purpose is to take away items we physically received ingame, but have not received from AP | ||||||
|     async def remove_physical_items(self, ctx: "BizHawkClientContext"): |     async def remove_physical_items(self, ctx: "BizHawkClientContext"): | ||||||
|  |         ram_addr_dict: dict[int, list[int]] = {} | ||||||
|  |  | ||||||
|         list_recv_itemids: list[int] = [netItem.item for netItem in ctx.items_received] |         list_recv_itemids: list[int] = [netItem.item for netItem in ctx.items_received] | ||||||
|         items_to_check: dict[str, GrinchItemData] = {**GADGETS_TABLE} #, **SLEIGH_PARTS_TABLE |         items_to_check: dict[str, GrinchItemData] = {**GADGETS_TABLE} #, **SLEIGH_PARTS_TABLE | ||||||
|         heart_count = len(list(item_id for item_id in list_recv_itemids if item_id == 42570)) |         heart_count = len(list(item_id for item_id in list_recv_itemids if item_id == 42570)) | ||||||
|         heart_item_data = ALL_ITEMS_TABLE["Heart of Stone"] |         heart_item_data = ALL_ITEMS_TABLE["Heart of Stone"] | ||||||
|         await self.update_and_validate_address(ctx, heart_item_data.update_ram_addr[0].ram_address, min(heart_count, 4), 1) |         ram_addr_dict[heart_item_data.update_ram_addr[0].ram_address] = [min(heart_count, 4), 1] | ||||||
|  |  | ||||||
|         # Setting Who Lake Mission Count back to 0 to prevent warping after completing 3 missions |         # Setting mission count for all accesses back to 0 to prevent warping/unlocking after completing 3 missions | ||||||
|         await self.update_and_validate_address(ctx,0x0100F0, 0, 4) |         ram_addr_dict[0x0100F0] = [0, 4] | ||||||
|  |  | ||||||
|         for (item_name, item_data) in items_to_check.items(): |         for (item_name, item_data) in items_to_check.items(): | ||||||
|             # If item is an event or already been received, ignore. |             # If item is an event or already been received, ignore. | ||||||
| @@ -207,42 +270,73 @@ class GrinchClient(BizHawkClient): | |||||||
|             for addr_to_update in item_data.update_ram_addr: |             for addr_to_update in item_data.update_ram_addr: | ||||||
|                 is_binary = True if not addr_to_update.binary_bit_pos is None else False |                 is_binary = True if not addr_to_update.binary_bit_pos is None else False | ||||||
|                 if is_binary: |                 if is_binary: | ||||||
|                     current_bin_value = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( |                     if addr_to_update.ram_address in ram_addr_dict.keys(): | ||||||
|                         addr_to_update.ram_address, addr_to_update.bit_size, "MainRAM")]))[0], "little") |                         current_bin_value = ram_addr_dict[addr_to_update.ram_address][0] | ||||||
|  |                     else: | ||||||
|  |                         current_bin_value = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( | ||||||
|  |                             addr_to_update.ram_address, addr_to_update.bit_size, "MainRAM")]))[0], "little") | ||||||
|                     current_bin_value &= ~(1 << addr_to_update.binary_bit_pos) |                     current_bin_value &= ~(1 << addr_to_update.binary_bit_pos) | ||||||
|                     await self.update_and_validate_address(ctx, addr_to_update.ram_address, current_bin_value, 1) |                     ram_addr_dict[addr_to_update.ram_address] = [current_bin_value, 1] | ||||||
|                 else: |                 else: | ||||||
|                     await self.update_and_validate_address(ctx, addr_to_update.ram_address, 0, 1) |                     ram_addr_dict[addr_to_update.ram_address] = [0, addr_to_update.bit_size] | ||||||
|  |  | ||||||
|  |         await bizhawk.write(ctx.bizhawk_ctx, self.convert_dict_to_ram_list(ram_addr_dict)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def convert_dict_to_ram_list(self, addr_dict: dict[int, list[int]]) -> list[tuple[int, Sequence[int], str]]: | ||||||
|  |         addr_list_to_update: list[tuple[int, Sequence[int], str]] = [] | ||||||
|  |  | ||||||
|  |         for (key, val) in addr_dict.items(): | ||||||
|  |             addr_list_to_update.append((key, val[0].to_bytes(val[1], "little"), "MainRAM")) | ||||||
|  |  | ||||||
|  |         return addr_list_to_update | ||||||
|  |  | ||||||
|     # Removes the regional access until you actually received it from AP. |     # Removes the regional access until you actually received it from AP. | ||||||
|     async def constant_address_update(self, ctx: "BizHawkClientContext"): |     async def constant_address_update(self, ctx: "BizHawkClientContext"): | ||||||
|  |         ram_addr_dict: dict[int, list[int]] = {} | ||||||
|  |  | ||||||
|         list_recv_itemids: list[int] = [netItem.item for netItem in ctx.items_received] |         list_recv_itemids: list[int] = [netItem.item for netItem in ctx.items_received] | ||||||
|         items_to_check: dict[str, GrinchItemData] = {**KEYS_TABLE, **MISSION_ITEMS_TABLE} |         items_to_check: dict[str, GrinchItemData] = {**KEYS_TABLE, **MISSION_ITEMS_TABLE} | ||||||
|  |  | ||||||
|         for (item_name, item_data) in items_to_check.items(): |         for (item_name, item_data) in items_to_check.items(): | ||||||
|             # If item is an event or already been received, ignore. |             # If item is an event or already been received, ignore. | ||||||
|             if item_data.id is None or GrinchLocation.get_apid(item_data.id) in list_recv_itemids: |             if item_data.id is None: # or GrinchLocation.get_apid(item_data.id) in list_recv_itemids: | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             # This assumes we don't have the item so we must set all the data to 0 |             # This will either constantly update the item to ensure you still have it or take it away if you don't deserve it | ||||||
|             for addr_to_update in item_data.update_ram_addr: |             for addr_to_update in item_data.update_ram_addr: | ||||||
|                 is_binary = True if not addr_to_update.binary_bit_pos is None else False |                 is_binary = True if not addr_to_update.binary_bit_pos is None else False | ||||||
|                 if is_binary: |                 if is_binary: | ||||||
|                     current_bin_value = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( |                     if addr_to_update.ram_address in ram_addr_dict.keys(): | ||||||
|                         addr_to_update.ram_address, addr_to_update.bit_size, "MainRAM")]))[0], "little") |                         current_bin_value = ram_addr_dict[addr_to_update.ram_address][0] | ||||||
|                     current_bin_value &= ~(1 << addr_to_update.binary_bit_pos) |                     else: | ||||||
|                     await self.update_and_validate_address(ctx, addr_to_update.ram_address, current_bin_value, 1) |                         current_bin_value = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( | ||||||
|  |                             addr_to_update.ram_address, addr_to_update.bit_size, "MainRAM")]))[0], "little") | ||||||
|  |                     if GrinchLocation.get_apid(item_data.id) in list_recv_itemids: | ||||||
|  |                         current_bin_value |= (1 << addr_to_update.binary_bit_pos) | ||||||
|  |                     else: | ||||||
|  |                         current_bin_value &= ~(1 << addr_to_update.binary_bit_pos) | ||||||
|  |  | ||||||
|  |                     ram_addr_dict[addr_to_update.ram_address] = [current_bin_value, 1] | ||||||
|                 else: |                 else: | ||||||
|                     await self.update_and_validate_address(ctx, addr_to_update.ram_address, 0, 1) |                     if GrinchLocation.get_apid(item_data.id) in list_recv_itemids: | ||||||
|  |                         ram_addr_dict[addr_to_update.ram_address] = [addr_to_update.value, addr_to_update.bit_size] | ||||||
|  |                     else: | ||||||
|  |                         ram_addr_dict[addr_to_update.ram_address] = [0, addr_to_update.bit_size] | ||||||
|  |  | ||||||
|  |         await bizhawk.write(ctx.bizhawk_ctx, self.convert_dict_to_ram_list(ram_addr_dict)) | ||||||
|  |  | ||||||
|     async def ingame_checker(self, ctx: "BizHawkClientContext"): |     async def ingame_checker(self, ctx: "BizHawkClientContext"): | ||||||
|         from CommonClient import logger |         from CommonClient import logger | ||||||
|  |  | ||||||
|         ingame_map_id = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( |         ingame_map_id = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( | ||||||
|             0x010000, 1, "MainRAM")]))[0], "little") |             0x010000, 1, "MainRAM")]))[0], "little") | ||||||
|  |         initial_cutscene_checker = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( | ||||||
|  |             0x010094, 1, "MainRAM")]))[0], "little") | ||||||
|  |  | ||||||
|         #If not in game or at a menu, or loading the publisher logos |         #If not in game or at a menu, or loading the publisher logos | ||||||
|         if ingame_map_id <= 0x04 or ingame_map_id >= 0x35: |         if ingame_map_id <= 0x04 or ingame_map_id >= 0x35: | ||||||
|  |             self.ingame_log = False | ||||||
|             return False |             return False | ||||||
|  |  | ||||||
|         #If grinch has changed maps |         #If grinch has changed maps | ||||||
| @@ -252,6 +346,7 @@ class GrinchClient(BizHawkClient): | |||||||
|                 # Reset our demo mode checker just in case the game is in demo mode. |                 # Reset our demo mode checker just in case the game is in demo mode. | ||||||
|                 self.demo_mode_buffer = 0 |                 self.demo_mode_buffer = 0 | ||||||
|                 self.ingame_log = False |                 self.ingame_log = False | ||||||
|  |             if initial_cutscene_checker != 1: | ||||||
|                 return False |                 return False | ||||||
|  |  | ||||||
|             # Update the previous map we were on to be the current map. |             # Update the previous map we were on to be the current map. | ||||||
| @@ -275,13 +370,67 @@ class GrinchClient(BizHawkClient): | |||||||
|  |  | ||||||
|     async def option_handler(self, ctx: "BizHawkClientContext"): |     async def option_handler(self, ctx: "BizHawkClientContext"): | ||||||
|         if self.loc_unlimited_eggs: |         if self.loc_unlimited_eggs: | ||||||
|             max_eggs: int = 200 |             await bizhawk.write(ctx.bizhawk_ctx, [(EGG_COUNT_ADDR, MAX_EGGS.to_bytes(2,"little"), "MainRAM")]) | ||||||
|             await bizhawk.write(ctx.bizhawk_ctx, [(0x010058, max_eggs.to_bytes(2,"little"), "MainRAM")]) |  | ||||||
|  |  | ||||||
|     async def update_and_validate_address(self, ctx: "BizHawkClientContext", address_to_validate: int, expected_value: int, byte_size: int): |     async def ring_link_output(self, ctx: "BizHawkClientContext"): | ||||||
|         await bizhawk.write(ctx.bizhawk_ctx, [(address_to_validate, expected_value.to_bytes(byte_size, "little"), "MainRAM")]) |         from CommonClient import logger | ||||||
|         current_value = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [(address_to_validate, byte_size, "MainRAM")]))[0], "little") |         while self.send_ring_link and ctx.slot: | ||||||
|         if not current_value == expected_value: |  | ||||||
|             if address_to_validate == 0x010000 or address_to_validate == 0x08FB94: # TODO Temporairly skips teleportation addresses; to be changed later on. |             try: | ||||||
|                 return |                 current_egg_count = int.from_bytes( | ||||||
|             raise Exception("Unable to update address as expected. Address: "+ str(address_to_validate)+"; Expected Value: "+str(expected_value)) |                     (await bizhawk.read(ctx.bizhawk_ctx, [(EGG_COUNT_ADDR, EGG_ADDR_BYTESIZE, "MainRAM")]))[0], "little") | ||||||
|  |  | ||||||
|  |                 if (current_egg_count - self.previous_egg_count) != 0: | ||||||
|  |                     msg = { | ||||||
|  |                         "cmd": "Bounce", | ||||||
|  |                         "data": { | ||||||
|  |                             "time": time.time(), | ||||||
|  |                             "source": self.unique_client_id, | ||||||
|  |                             "amount": current_egg_count - self.previous_egg_count | ||||||
|  |                         }, | ||||||
|  |                         "tags": ["RingLink"] | ||||||
|  |                     } | ||||||
|  |                     await ctx.send_msgs([msg]) | ||||||
|  |                     self.previous_egg_count = current_egg_count | ||||||
|  |                     # logger.info(f"RingLink: You sent {str(current_egg_count - self.previous_egg_count)} rotten eggs.") | ||||||
|  |                 await asyncio.sleep(0.1) | ||||||
|  |             except Exception as ex: | ||||||
|  |                 logger.error("While monitoring grinch's egg count ingame, an error occurred. Details:"+ str(ex)) | ||||||
|  |                 self.send_ring_link = False | ||||||
|  |  | ||||||
|  |         if not ctx.slot: | ||||||
|  |             logger.info("You must be connected to the multi-world in order for RingLink to work properly.") | ||||||
|  |  | ||||||
|  |     async def ring_link_input(self, egg_amount: int, ctx: "BizHawkClientContext"): | ||||||
|  |         from CommonClient import logger | ||||||
|  |         game_egg_count = int.from_bytes( | ||||||
|  |             (await bizhawk.read(ctx.bizhawk_ctx, [(EGG_COUNT_ADDR, EGG_ADDR_BYTESIZE, "MainRAM")]))[0], "little") | ||||||
|  |         non_neg_eggs = game_egg_count + egg_amount if game_egg_count + egg_amount > 0 else 0 | ||||||
|  |         current_egg_count = min(non_neg_eggs, MAX_EGGS) | ||||||
|  |         await bizhawk.write(ctx.bizhawk_ctx, [(EGG_COUNT_ADDR, | ||||||
|  |             int(current_egg_count).to_bytes(EGG_ADDR_BYTESIZE, "little"), "MainRAM")]) | ||||||
|  |         self.previous_egg_count = current_egg_count | ||||||
|  |         # logger.info(f"RingLink: You received {str(egg_amount)} rotten eggs.") | ||||||
|  |  | ||||||
|  |     def _get_uuid(self) -> int: | ||||||
|  |         string_id = str(uuid.uuid4()) | ||||||
|  |         uid: int = 0 | ||||||
|  |         for char in string_id: | ||||||
|  |             uid += ord(char) | ||||||
|  |         return uid | ||||||
|  |  | ||||||
|  | def _cmd_ringlink(self): | ||||||
|  |     """Toggle ringling from client. Overrides default setting.""" | ||||||
|  |     if not self.ctx.slot: | ||||||
|  |         return | ||||||
|  |     Utils.async_start(_update_ring_link(self.ctx, not "RingLink" in self.ctx.tags), name="Update RingLink") | ||||||
|  |  | ||||||
|  | async def _update_ring_link(ctx: "BizHawkClientContext", ring_link: bool): | ||||||
|  |     """Helper function to set Ring Link connection tag on/off and update the connection if already connected.""" | ||||||
|  |     old_tags = copy.deepcopy(ctx.tags) | ||||||
|  |     if ring_link: | ||||||
|  |         ctx.tags.add("RingLink") | ||||||
|  |     else: | ||||||
|  |         ctx.tags -= {"RingLink"} | ||||||
|  |     if old_tags != ctx.tags and ctx.server and not ctx.server.socket.closed: | ||||||
|  |         await ctx.send_msgs([{"cmd": "ConnectUpdate", "tags": ctx.tags}]) | ||||||
| @@ -5,11 +5,10 @@ from BaseClasses import Item | |||||||
| from BaseClasses import ItemClassification as IC #IC can be any name, saves having to type the whole word in code | from BaseClasses import ItemClassification as IC #IC can be any name, saves having to type the whole word in code | ||||||
|  |  | ||||||
| class GrinchItemData(NamedTuple): | class GrinchItemData(NamedTuple): | ||||||
|     item_group: str #arbituary that can be whatever it can be, basically the field/property for item groups |     item_group: list[str] #arbituary that can be whatever it can be, basically the field/property for item groups | ||||||
|     id: Optional[int] |     id: Optional[int] | ||||||
|     classification: IC |     classification: IC | ||||||
|     update_ram_addr: list[GrinchRamData] |     update_ram_addr: list[GrinchRamData] | ||||||
|     second_item_group: Optional[str] = None |  | ||||||
|  |  | ||||||
| class GrinchItem(Item): | class GrinchItem(Item): | ||||||
|     game: str = "The Grinch" |     game: str = "The Grinch" | ||||||
| @@ -32,24 +31,54 @@ def get_item_names_per_category() -> dict[str, set[str]]: | |||||||
|     categories: dict[str, set[str]] = {} |     categories: dict[str, set[str]] = {} | ||||||
|  |  | ||||||
|     for name, data in ALL_ITEMS_TABLE.items(): |     for name, data in ALL_ITEMS_TABLE.items(): | ||||||
|         categories.setdefault(data.item_group, set()).add(name) |         for group in data.item_group:  # iterate over each category | ||||||
|  |             categories.setdefault(group, set()).add(name) | ||||||
|  |  | ||||||
|     return categories |     return categories | ||||||
|  |  | ||||||
|  | REL: str = "Rotten Egg Launcher" | ||||||
|  | RS: str = "Rocket Spring" | ||||||
|  | SS: str = "Slime Shooter" | ||||||
|  | OCD: str = "Octopus Climbing Device" | ||||||
|  | MM: str = "Marine Mobile" | ||||||
|  | GC: str = "Grinch Copter" | ||||||
|  | WV: str = "Whoville Vacuum Tube" | ||||||
|  | WF: str = "Who Forest Vacuum Tube" | ||||||
|  | WD: str = "Who Dump Vacuum Tube" | ||||||
|  | WL: str = "Who Lake Vacuum Tube" | ||||||
|  | VT: str = "Progressive Vacuum Tube" | ||||||
|  | PC: str = "Pancake" | ||||||
|  | SR: str = "Sleigh Room Key" | ||||||
|  | BB: str = "Bad Breath" | ||||||
|  | SZ: str = "Seize" | ||||||
|  | MX: str = "Max" | ||||||
|  | SN: str = "Sneak" | ||||||
|  | WC: str = "Who Cloak" | ||||||
|  | PB: str = "Painting Bucket" | ||||||
|  | SC: str = "Scissors" | ||||||
|  | GB: str = "Glue Bucket" | ||||||
|  | CCAC: str = "Cable Car Access Card" | ||||||
|  | DRL: str = "Drill" | ||||||
|  | RP: str = "Rope" | ||||||
|  | HK: str = "Hook" | ||||||
|  | ST: str = "Sculpting Tools" | ||||||
|  | HMR: str = "Hammer" | ||||||
|  | SCL: str = "Scout Clothes" | ||||||
|  |  | ||||||
| #Gadgets | #Gadgets | ||||||
| #All gadgets require at least 4 different blueprints to be unlocked in the computer in Mount Crumpit. | #All gadgets require at least 4 different blueprints to be unlocked in the computer in Mount Crumpit. | ||||||
| GADGETS_TABLE: dict[str, GrinchItemData] = { | GADGETS_TABLE: dict[str, GrinchItemData] = { | ||||||
|     "Binoculars": GrinchItemData("Gadgets", 100, IC.useful, |     "Binoculars": GrinchItemData(["Gadgets"], 100, IC.useful, | ||||||
|         [GrinchRamData(0x0102B6, value=0x40), GrinchRamData(0x0102B7, value=0x41), |         [GrinchRamData(0x0102B6, value=0x40), GrinchRamData(0x0102B7, value=0x41), | ||||||
|         GrinchRamData(0x0102B8, value=0x44), GrinchRamData(0x0102B9, value=0x45), |         GrinchRamData(0x0102B8, value=0x44), GrinchRamData(0x0102B9, value=0x45), | ||||||
|         # GrinchRamData(0x0100BC, binary_bit_pos=0) |         # GrinchRamData(0x0100BC, binary_bit_pos=0) | ||||||
|          ]), |          ]), | ||||||
|     "Rotten Egg Launcher": GrinchItemData("Gadgets", 101, IC.progression, |     "Rotten Egg Launcher": GrinchItemData(["Gadgets"], 101, IC.progression, | ||||||
|         [GrinchRamData(0x0102BA, value=0x40), GrinchRamData(0x0102BB, value=0x41), |         [GrinchRamData(0x0102BA, value=0x40), GrinchRamData(0x0102BB, value=0x41), | ||||||
|         GrinchRamData(0x0102BC, value=0x44), GrinchRamData(0x0102BD, value=0x45), |         GrinchRamData(0x0102BC, value=0x44), GrinchRamData(0x0102BD, value=0x45), | ||||||
|         # GrinchRamData(0x0100BC, binary_bit_pos=1) |         # GrinchRamData(0x0100BC, binary_bit_pos=1) | ||||||
|          ]), |          ]), | ||||||
|     "Rocket Spring": GrinchItemData("Gadgets", 102, IC.progression, |     "Rocket Spring": GrinchItemData(["Gadgets"], 102, IC.progression, | ||||||
|         [GrinchRamData(0x0102BE, value=0x40), GrinchRamData(0x0102BF, value=0x41), |         [GrinchRamData(0x0102BE, value=0x40), GrinchRamData(0x0102BF, value=0x41), | ||||||
|         GrinchRamData(0x0102C0, value=0x42), GrinchRamData(0x0102C1, value=0x44), |         GrinchRamData(0x0102C0, value=0x42), GrinchRamData(0x0102C1, value=0x44), | ||||||
|         GrinchRamData(0x0102C2, value=0x45), GrinchRamData(0x0102C3, value=0x46), |         GrinchRamData(0x0102C2, value=0x45), GrinchRamData(0x0102C3, value=0x46), | ||||||
| @@ -57,7 +86,7 @@ GADGETS_TABLE: dict[str, GrinchItemData] = { | |||||||
|         GrinchRamData(0x0102C6, value=0x4A), |         GrinchRamData(0x0102C6, value=0x4A), | ||||||
|          # GrinchRamData(0x0100BC, binary_bit_pos=2) |          # GrinchRamData(0x0100BC, binary_bit_pos=2) | ||||||
|          ]), |          ]), | ||||||
|     "Slime Shooter": GrinchItemData("Gadgets", 103, IC.progression, |     "Slime Shooter": GrinchItemData(["Gadgets", "Slime Gun"], 103, IC.progression, | ||||||
|         [GrinchRamData(0x0102C7, value=0x40), GrinchRamData(0x0102C8, value=0x41), |         [GrinchRamData(0x0102C7, value=0x40), GrinchRamData(0x0102C8, value=0x41), | ||||||
|         GrinchRamData(0x0102C9, value=0x42), GrinchRamData(0x0102CA, value=0x44), |         GrinchRamData(0x0102C9, value=0x42), GrinchRamData(0x0102CA, value=0x44), | ||||||
|         GrinchRamData(0x0102CB, value=0x45), GrinchRamData(0x0102CC, value=0x46), |         GrinchRamData(0x0102CB, value=0x45), GrinchRamData(0x0102CC, value=0x46), | ||||||
| @@ -65,7 +94,7 @@ GADGETS_TABLE: dict[str, GrinchItemData] = { | |||||||
|         GrinchRamData(0x0102CF, value=0x4A), |         GrinchRamData(0x0102CF, value=0x4A), | ||||||
|          # GrinchRamData(0x0100BC, binary_bit_pos=3) |          # GrinchRamData(0x0100BC, binary_bit_pos=3) | ||||||
|          ]), |          ]), | ||||||
|     "Octopus Climbing Device": GrinchItemData("Gadgets", 104, IC.progression, |     "Octopus Climbing Device": GrinchItemData(["Gadgets"], 104, IC.progression, | ||||||
|         [GrinchRamData(0x0102D0, value=0x40), GrinchRamData(0x0102D1, value=0x41), |         [GrinchRamData(0x0102D0, value=0x40), GrinchRamData(0x0102D1, value=0x41), | ||||||
|         GrinchRamData(0x0102D2, value=0x42), GrinchRamData(0x0102D3, value=0x44), |         GrinchRamData(0x0102D2, value=0x42), GrinchRamData(0x0102D3, value=0x44), | ||||||
|         GrinchRamData(0x0102D4, value=0x45), GrinchRamData(0x0102D5, value=0x46), |         GrinchRamData(0x0102D4, value=0x45), GrinchRamData(0x0102D5, value=0x46), | ||||||
| @@ -73,7 +102,7 @@ GADGETS_TABLE: dict[str, GrinchItemData] = { | |||||||
|         GrinchRamData(0x0102D8, value=0x4A), |         GrinchRamData(0x0102D8, value=0x4A), | ||||||
|          # GrinchRamData(0x0100BC, binary_bit_pos=4) |          # GrinchRamData(0x0100BC, binary_bit_pos=4) | ||||||
|          ]), |          ]), | ||||||
|     "Marine Mobile": GrinchItemData("Gadgets", 105, IC.progression, |     "Marine Mobile": GrinchItemData(["Gadgets"], 105, IC.progression, | ||||||
|         [GrinchRamData(0x0102D9, value=0x40), GrinchRamData(0x0102DA, value=0x41), |         [GrinchRamData(0x0102D9, value=0x40), GrinchRamData(0x0102DA, value=0x41), | ||||||
|         GrinchRamData(0x0102DB, value=0x42), GrinchRamData(0x0102DC, value=0x43), |         GrinchRamData(0x0102DB, value=0x42), GrinchRamData(0x0102DC, value=0x43), | ||||||
|         GrinchRamData(0x0102DD, value=0x44), GrinchRamData(0x0102DE, value=0x45), |         GrinchRamData(0x0102DD, value=0x44), GrinchRamData(0x0102DE, value=0x45), | ||||||
| @@ -84,7 +113,7 @@ GADGETS_TABLE: dict[str, GrinchItemData] = { | |||||||
|         GrinchRamData(0x0102E7, value=0x4E), GrinchRamData(0x0102E8, value=0x4F), |         GrinchRamData(0x0102E7, value=0x4E), GrinchRamData(0x0102E8, value=0x4F), | ||||||
|         # GrinchRamData(0x0100BC, binary_bit_pos=5) |         # GrinchRamData(0x0100BC, binary_bit_pos=5) | ||||||
|          ]), |          ]), | ||||||
|     "Grinch Copter": GrinchItemData("Gadgets", 106, IC.progression, |     "Grinch Copter": GrinchItemData(["Gadgets"], 106, IC.progression, | ||||||
|         [GrinchRamData(0x0102E9, value=0x40), GrinchRamData(0x0102EA, value=0x41), |         [GrinchRamData(0x0102E9, value=0x40), GrinchRamData(0x0102EA, value=0x41), | ||||||
|         GrinchRamData(0x0102EB, value=0x42), GrinchRamData(0x0102EC, value=0x43), |         GrinchRamData(0x0102EB, value=0x42), GrinchRamData(0x0102EC, value=0x43), | ||||||
|         GrinchRamData(0x0102ED, value=0x44), GrinchRamData(0x0102EE, value=0x45), |         GrinchRamData(0x0102ED, value=0x44), GrinchRamData(0x0102EE, value=0x45), | ||||||
| @@ -99,119 +128,120 @@ GADGETS_TABLE: dict[str, GrinchItemData] = { | |||||||
|  |  | ||||||
| #Mission Specific Items | #Mission Specific Items | ||||||
| MISSION_ITEMS_TABLE: dict[str, GrinchItemData] = { | MISSION_ITEMS_TABLE: dict[str, GrinchItemData] = { | ||||||
|     "Who Cloak": GrinchItemData("Mission Specific Items", 200, IC.progression, |     "Who Cloak": GrinchItemData(["Mission Specific Items", "Useful Items"], 200, IC.progression, | ||||||
|         [GrinchRamData(0x0101F9, binary_bit_pos=0)], second_item_group="Useful Items"), |         [GrinchRamData(0x0101F9, binary_bit_pos=0)]), | ||||||
|     "Painting Bucket": GrinchItemData("Mission Specific Items", 201, IC.progression_deprioritized, |     "Painting Bucket": GrinchItemData(["Mission Specific Items", "Useful Items"], 201, IC.progression_deprioritized, | ||||||
|         [GrinchRamData(0x0101F9, binary_bit_pos=1)], second_item_group="Useful Items"), |         [GrinchRamData(0x0101F9, binary_bit_pos=1)]), | ||||||
|     "Scissors": GrinchItemData("Mission Specific Items", 202, IC.progression_deprioritized, |     "Scissors": GrinchItemData(["Mission Specific Items", "Useful Items"], 202, IC.progression_deprioritized, | ||||||
|         [GrinchRamData(0x0101F9, binary_bit_pos=6), GrinchRamData(0x0100C2, binary_bit_pos=1)], |         [GrinchRamData(0x0101F9, binary_bit_pos=6), GrinchRamData(0x0100C2, binary_bit_pos=1)]), | ||||||
|         second_item_group="Useful Items"), |     "Glue Bucket": GrinchItemData(["Mission Specific Items", "Useful Items"], 203, IC.progression_deprioritized, | ||||||
|     "Glue Bucket": GrinchItemData("Mission Specific Items", 203, IC.progression_deprioritized, |         [GrinchRamData(0x0101F9, binary_bit_pos=4)]), | ||||||
|         [GrinchRamData(0x0101F9, binary_bit_pos=4)], second_item_group="Useful Items"), |     "Cable Car Access Card": GrinchItemData(["Mission Specific Items", "Useful Items"], 204, IC.progression, | ||||||
|     "Cable Car Access Card": GrinchItemData("Mission Specific Items", 204, IC.progression, |         [GrinchRamData(0x0101F9, binary_bit_pos=5)]), | ||||||
|         [GrinchRamData(0x0101F9, binary_bit_pos=5)], second_item_group="Useful Items"), |     "Drill": GrinchItemData(["Mission Specific Items", "Useful Items"], 205, IC.progression_deprioritized, | ||||||
|     "Drill": GrinchItemData("Mission Specific Items", 205, IC.progression_deprioritized, |         [GrinchRamData(0x0101FA, binary_bit_pos=2)]), | ||||||
|         [GrinchRamData(0x0101FA, binary_bit_pos=2)], second_item_group="Useful Items"), |     "Rope": GrinchItemData(["Mission Specific Items", "Useful Items"], 206, IC.progression_deprioritized, | ||||||
|     "Rope": GrinchItemData("Mission Specific Items", 206, IC.progression_deprioritized, |         [GrinchRamData(0x0101FA, binary_bit_pos=1)]), | ||||||
|         [GrinchRamData(0x0101FA, binary_bit_pos=1)], second_item_group="Useful Items"), |     "Hook": GrinchItemData(["Mission Specific Items", "Useful Items"], 207, IC.progression_deprioritized, | ||||||
|     "Hook": GrinchItemData("Mission Specific Items", 207, IC.progression_deprioritized, |         [GrinchRamData(0x0101FA, binary_bit_pos=0)]), | ||||||
|         [GrinchRamData(0x0101FA, binary_bit_pos=0)], second_item_group="Useful Items"), |     "Sculpting Tools": GrinchItemData(["Mission Specific Items", "Useful Items"], 208, IC.progression_deprioritized, | ||||||
|     "Sculpting Tools": GrinchItemData("Mission Specific Items", 208, IC.progression_deprioritized, |         [GrinchRamData(0x0101F9, binary_bit_pos=2)]), | ||||||
|         [GrinchRamData(0x0101F9, binary_bit_pos=2)], second_item_group="Useful Items"), |     "Hammer": GrinchItemData(["Mission Specific Items", "Useful Items"], 209, IC.progression_deprioritized, | ||||||
|     "Hammer": GrinchItemData("Mission Specific Items", 209, IC.progression_deprioritized, |         [GrinchRamData(0x0101F9, binary_bit_pos=3)]), | ||||||
|         [GrinchRamData(0x0101F9, binary_bit_pos=3)], second_item_group="Useful Items"), |     "Scout Clothes": GrinchItemData(["Mission Specific Items", "Useful Items"], 210, IC.progression, | ||||||
|     "Scout Clothes": GrinchItemData("Mission Specific Items", 210, IC.progression, |         [GrinchRamData(0x0101F9, binary_bit_pos=7)]) | ||||||
|         [GrinchRamData(0x0101F9, binary_bit_pos=7)], second_item_group="Useful Items") |  | ||||||
| } | } | ||||||
|  |  | ||||||
| #Sleigh Parts | #Sleigh Parts | ||||||
| # SLEIGH_PARTS_TABLE: dict[str, GrinchItemData] = { | # SLEIGH_PARTS_TABLE: dict[str, GrinchItemData] = { | ||||||
| #     "Exhaust Pipes": GrinchItemData("Sleigh Parts", 300, IC.progression_skip_balancing, | #     "Exhaust Pipes": GrinchItemData(["Sleigh Parts"], 300, IC.progression_skip_balancing, | ||||||
| #         [GrinchRamData(0x0101FB, binary_bit_pos=2)]), | #         [GrinchRamData(0x0101FB, binary_bit_pos=2)]), | ||||||
| #     "GPS": GrinchItemData("Sleigh Parts", 301, IC.useful, | #     "GPS": GrinchItemData(["Sleigh Parts"], 301, IC.useful, | ||||||
| #         [GrinchRamData(0x0101FB, binary_bit_pos=5)]), | #         [GrinchRamData(0x0101FB, binary_bit_pos=5)]), | ||||||
| #     "Tires": GrinchItemData("Sleigh Parts", 302, IC.progression_skip_balancing, | #     "Tires": GrinchItemData(["Sleigh Parts"], 302, IC.progression_skip_balancing, | ||||||
| #         [GrinchRamData(0x0101FB, binary_bit_pos=4)]), | #         [GrinchRamData(0x0101FB, binary_bit_pos=4)]), | ||||||
| #     "Skis": GrinchItemData("Sleigh Parts", 303, IC.progression_skip_balancing, | #     "Skis": GrinchItemData(["Sleigh Parts"], 303, IC.progression_skip_balancing, | ||||||
| #         [GrinchRamData(0x0101FB, binary_bit_pos=3)]), | #         [GrinchRamData(0x0101FB, binary_bit_pos=3)]), | ||||||
| #     "Twin-End Tuba": GrinchItemData("Sleigh Parts", 304, IC.progression_skip_balancing, | #     "Twin-End Tuba": GrinchItemData(["Sleigh Parts"], 304, IC.progression_skip_balancing, | ||||||
| #         [GrinchRamData(0x0101FB, binary_bit_pos=6)]) | #         [GrinchRamData(0x0101FB, binary_bit_pos=6)]) | ||||||
| # } | # } | ||||||
|  |  | ||||||
| #Access Keys | #Access Keys | ||||||
| KEYS_TABLE: dict[str, GrinchItemData] = { | KEYS_TABLE: dict[str, GrinchItemData] = { | ||||||
|     # "Whoville Vacuum Access": GrinchItemData("Vacuum Access", 400, IC.progression, |     "Whoville Vacuum Tube": GrinchItemData(["Vacuum Tubes"], 400, IC.progression, | ||||||
|     #     [GrinchRamData()]), |         [GrinchRamData(0x010200, binary_bit_pos=1)]), | ||||||
|     "Who Forest Vacuum Access": GrinchItemData("Vacuum Access", 401, IC.progression, |     "Who Forest Vacuum Tube": GrinchItemData(["Vacuum Tubes"], 401, IC.progression, | ||||||
|         [GrinchRamData(0x0100AA, binary_bit_pos=2)]), |         [GrinchRamData(0x0100AA, binary_bit_pos=2)]), | ||||||
|     "Who Dump Vacuum Access": GrinchItemData("Vacuum Access", 402, IC.progression, |     "Who Dump Vacuum Tube": GrinchItemData(["Vacuum Tubes"], 402, IC.progression, | ||||||
|         [GrinchRamData(0x0100AA, binary_bit_pos=3)]), |         [GrinchRamData(0x0100AA, binary_bit_pos=3)]), | ||||||
|     "Who Lake Vacuum Access": GrinchItemData("Vacuum Access", 403, IC.progression, |     "Who Lake Vacuum Tube": GrinchItemData(["Vacuum Tubes"], 403, IC.progression, | ||||||
|         [GrinchRamData(0x0100AA, binary_bit_pos=4)]), |         [GrinchRamData(0x0100AA, binary_bit_pos=4)]), | ||||||
|     # "Progressive Vacuum Access": GrinchItemData("Vacuum Access", 404, IC.progression, |     # "Progressive Vacuum Tube": GrinchItemData(["Vacuum Tubes"], 404, IC.progression, | ||||||
|     #     [GrinchRamData()]), |     #     [GrinchRamData()]), | ||||||
|     # "Spin N' Win Door Unlock": GrinchItemData("Supadow Door Unlocks", 405, IC.progression, |     # "Spin N' Win Door Unlock": GrinchItemData(["Supadow Door Unlocks"], 405, IC.progression, | ||||||
|     #     [GrinchRamData()]), |     #     [GrinchRamData()]), | ||||||
|     # "Dankamania Door Unlock": GrinchItemData("Supadow Door Unlocks", 406, IC.progression, |     # "Dankamania Door Unlock": GrinchItemData(["Supadow Door Unlocks"], 406, IC.progression, | ||||||
|     #     [GrinchRamData()]), |     #     [GrinchRamData()]), | ||||||
|     # "The Copter Race Contest Door Unlock": GrinchItemData("Supadow Door Unlocks", 407, IC.progression, |     # "The Copter Race Contest Door Unlock": GrinchItemData("Supadow Door Unlocks", 407, IC.progression, | ||||||
|     #     [GrinchRamData()]), |     #     [GrinchRamData()]), | ||||||
|     # "Progressive Supadow Door Unlock": GrinchItemData("Supadow Door Unlocks", 408, IC.progression, |     # "Progressive Supadow Door Unlock": GrinchItemData("Supadow Door Unlocks", 408, IC.progression, | ||||||
|     #     [GrinchRamData()]), |     #     [GrinchRamData()]), | ||||||
|     # "Bike Race Access": GrinchItemData("Supadow Door Unlocks", 409, IC.progression, |     # "Bike Race Access": GrinchItemData(["Supadow Door Unlocks", 409, IC.progression, | ||||||
|     #     [GrinchRamData()]) |     #     [GrinchRamData()]) | ||||||
|     "Sleigh Room Key": GrinchItemData("Sleigh Room", 410, IC.progression, |     "Sleigh Room Key": GrinchItemData(["Sleigh Room"], 410, IC.progression, | ||||||
|         [GrinchRamData(0x010200, binary_bit_pos=6), GrinchRamData(0x0100AA, binary_bit_pos=5)]) |         [GrinchRamData(0x010200, binary_bit_pos=6), GrinchRamData(0x0100AA, binary_bit_pos=5)]) | ||||||
| } | } | ||||||
|  |  | ||||||
| #Misc Items | #Misc Items | ||||||
| MISC_ITEMS_TABLE: dict[str, GrinchItemData] = { | MISC_ITEMS_TABLE: dict[str, GrinchItemData] = { | ||||||
|     # This item may not function properly if you receive it during a loading screen or in Mount Crumpit |     # This item may not function properly if you receive it during a loading screen or in Mount Crumpit | ||||||
|     # "Fully Healed Grinch": GrinchItemData("Health Items", 500, IC.filler, |     # "Fully Healed Grinch": GrinchItemData(["Health Items", "Filler"], 500, IC.filler, | ||||||
|     #     [GrinchRamData(0x0E8FDC, value=120)]), |     #     [GrinchRamData(0x0E8FDC, value=120)]), | ||||||
|     "5 Rotten Eggs": GrinchItemData("Rotten Egg Bundles", 502, IC.filler, |     "5 Rotten Eggs": GrinchItemData(["Rotten Egg Bundles", "Filler"], 502, IC.filler, | ||||||
|         [GrinchRamData(0x010058, value=5, update_existing_value=True, max_count=200, bit_size=2)]), |         [GrinchRamData(0x010058, value=5, update_existing_value=True, max_count=200, bit_size=2)]), | ||||||
|     "10 Rotten Eggs": GrinchItemData("Rotten Egg Bundles", 503, IC.filler, |     "10 Rotten Eggs": GrinchItemData(["Rotten Egg Bundles", "Filler"], 503, IC.filler, | ||||||
|         [GrinchRamData(0x010058, value=10, update_existing_value=True, max_count=200, bit_size=2)]), |         [GrinchRamData(0x010058, value=10, update_existing_value=True, max_count=200, bit_size=2)]), | ||||||
|     "20 Rotten Eggs": GrinchItemData("Rotten Egg Bundles", 504, IC.filler, |     "20 Rotten Eggs": GrinchItemData(["Rotten Egg Bundles", "Filler"], 504, IC.filler, | ||||||
|         [GrinchRamData(0x010058, value=20, update_existing_value=True, max_count=200, bit_size=2)]) |         [GrinchRamData(0x010058, value=20, update_existing_value=True, max_count=200, bit_size=2)]) | ||||||
| } | } | ||||||
|  |  | ||||||
| USEFUL_IC_TABLE: dict[str, GrinchItemData] = { | USEFUL_IC_TABLE: dict[str, GrinchItemData] = { | ||||||
|     "Heart of Stone": GrinchItemData("Health Items", 501, IC.useful, |     "Heart of Stone": GrinchItemData(["Health Items"], 501, IC.useful, | ||||||
|         [GrinchRamData(0x0100ED, value=1, update_existing_value=True, max_count=4)]) |         [GrinchRamData(0x0100ED, value=1, update_existing_value=True, max_count=4)]) | ||||||
| } | } | ||||||
|  |  | ||||||
| #Traps | #Traps | ||||||
| TRAPS_TABLE: dict[str, GrinchItemData] = { | TRAPS_TABLE: dict[str, GrinchItemData] = { | ||||||
| # alias to Ice Trap for traplink | # alias to Ice Trap for traplink | ||||||
|     # "Freeze Trap": GrinchItemData("Traps", 600, IC.trap, [GrinchRamData()]), |     # "Freeze Trap": GrinchItemData(["Traps"], 600, IC.trap, [GrinchRamData()]), | ||||||
|     # "Bee Trap": GrinchItemData("Traps", 601, IC.trap, [GrinchRamData()]), |     # "Bee Trap": GrinchItemData(["Traps"], 601, IC.trap, [GrinchRamData()]), | ||||||
|     # "Electrocution Trap": GrinchItemData("Traps", 602, IC.trap, [GrinchRamData()]), |     # "Electrocution Trap": GrinchItemData(["Traps"], 602, IC.trap, [GrinchRamData()]), | ||||||
| # alias to Slowness Trap for traplink | # alias to Slowness Trap for traplink | ||||||
|     # "Tip Toe Trap": GrinchItemData("Traps", 603, IC.trap, [GrinchRamData()]), |     # "Tip Toe Trap": GrinchItemData(["Traps"], 603, IC.trap, [GrinchRamData()]), | ||||||
| # This item may not function properly if you receive it during a loading screen or in Mount Crumpit | # This item may not function properly if you receive it during a loading screen or in Mount Crumpit | ||||||
| #     "Damage Trap": GrinchItemData("Traps", 604, IC.trap, [GrinchRamData(0x0E8FDC, value=-20, update_existing_value=True)]), | # alias to Exhaustion Trap | ||||||
|     "Depletion Trap": GrinchItemData("Traps", 605, IC.trap, [GrinchRamData(0x010058, value=0, bit_size=2)]), | #     "Damage Trap": GrinchItemData(["Traps"], 604, IC.trap, [GrinchRamData(0x0E8FDC, value=-20, update_existing_value=True)]), | ||||||
|     "Dump it to Crumpit": GrinchItemData("Traps", 606, IC.trap, #Alias to Home Trap for traplink |     "Depletion Trap": GrinchItemData(["Traps"], 605, IC.trap, [GrinchRamData(0x010058, value=0, bit_size=2)]), | ||||||
|         [GrinchRamData(0x010000, value=0x05), GrinchRamData(0x08FB94, value=1)]), |     "Dump it to Crumpit": GrinchItemData(["Traps"], 606, IC.trap, #Alias to Home Trap for traplink | ||||||
|  |         [GrinchRamData(0x010000, value=0x05), GrinchRamData(0x08FB94, value=1), GrinchRamData(0x0100B4, value=0)]), | ||||||
| #alias to Spring Trap for traplink | #alias to Spring Trap for traplink | ||||||
|     # "Rocket Spring Trap": GrinchItemData("Traps", 607, IC.trap, [GrinchRamData()]), |     # "Rocket Spring Trap": GrinchItemData(["Traps"], 607, IC.trap, [GrinchRamData()]), | ||||||
| #alias to Home Trap for traplink | #alias to Home Trap for traplink | ||||||
|     "Who sent me back?": GrinchItemData("Traps", 608, IC.trap, [GrinchRamData(0x08FB94, value=1)]), |     "Who sent me back?": GrinchItemData(["Traps"], 608, IC.trap, [GrinchRamData(0x08FB94, value=1)]), | ||||||
|     # "Cutscene Trap": GrinchItemData("Traps", 609, IC.trap, [GrinchRamData()]), |     # "Cutscene Trap": GrinchItemData(["Traps"], 609, IC.trap, [GrinchRamData()]), | ||||||
|     # "No Vac Trap": GrinchItemData("Traps", 610, IC.trap, [GrinchRamData(0x0102DA, value=0]), |     # "No Vac Trap": GrinchItemData(["Traps"], 610, IC.trap, [GrinchRamData(0x0102DA, value=0]), | ||||||
|     # "Invisible Trap": GrinchItemData("Traps", 611, IC.trap, [GrinchRamData(0x0102DA, value=0, bit_size=4)]) |     # "Invisible Trap": GrinchItemData(["Traps"], 611, IC.trap, [GrinchRamData(0x0102DA, value=0, bit_size=4)]) | ||||||
|     # "Child Trap": GrinchItemData("Traps", 612, IC.trap,[GrinchRamData()]) |     # "Child Trap": GrinchItemData(["Traps"], 612, IC.trap,[GrinchRamData()]) | ||||||
|  |     # "Disable Jump Trap": GrinchItemData(["Traps"], 613, IC.trap,[GrinchRamData(0x010026, binary_bit_pos=6)]) | ||||||
| } | } | ||||||
|  |  | ||||||
| #Movesets | #Movesets | ||||||
| # MOVES_TABLE: dict[str, GrinchItemData] = { | # MOVES_TABLE: dict[str, GrinchItemData] = { | ||||||
| #     "Bad Breath": GrinchItemData("Movesets", 700, IC.progression, [GrinchRamData(0x0100BB, binary_bit_pos=1)]), | #     "Bad Breath": GrinchItemData(["Movesets"], 700, IC.progression, [GrinchRamData(0x0100BB, binary_bit_pos=1)]), | ||||||
| #     "Pancake": GrinchItemData("Movesets", 701, IC.progression, [GrinchRamData(0x0100BB, binary_bit_pos=2)]), | #     "Pancake": GrinchItemData(["Movesets"], 701, IC.progression, [GrinchRamData(0x0100BB, binary_bit_pos=2)]), | ||||||
| #     "Push & Pull": GrinchItemData("Movesets", 702, IC.progression, [GrinchRamData(0x0100BB, binary_bit_pos=3)]), | #     "Push & Pull": GrinchItemData(["Movesets"], 702, IC.progression, [GrinchRamData(0x0100BB, binary_bit_pos=3)]), | ||||||
| #     "Max": GrinchItemData("Movesets", 703, IC.progression, [GrinchRamData(0x0100BB, binary_bit_pos=4)]), | #     "Max": GrinchItemData(["Movesets"], 703, IC.progression, [GrinchRamData(0x0100BB, binary_bit_pos=4)]), | ||||||
| #     "Tip Toe": GrinchItemData("Movesets", 704, IC.progression, [GrinchRamData(0x0100BB, binary_bit_pos=5)]) | #     "Tip Toe": GrinchItemData(["Movesets"], 704, IC.progression, [GrinchRamData(0x0100BB, binary_bit_pos=5)]) | ||||||
| # } | # } | ||||||
| #Double star combines all dictionaries from each individual list together | #Double star combines all dictionaries from each individual list together | ||||||
| ALL_ITEMS_TABLE: dict[str, GrinchItemData] = { | ALL_ITEMS_TABLE: dict[str, GrinchItemData] = { | ||||||
| @@ -228,12 +258,15 @@ ALL_ITEMS_TABLE: dict[str, GrinchItemData] = { | |||||||
| # Psuedocoding traplink table | # Psuedocoding traplink table | ||||||
| # BEE_TRAP_EQUIV = ["Army Trap", "Buyon Trap", "Ghost", "Gooey Bag", "OmoTrap", "Police Trap"] | # BEE_TRAP_EQUIV = ["Army Trap", "Buyon Trap", "Ghost", "Gooey Bag", "OmoTrap", "Police Trap"] | ||||||
| # ICE_TRAP_EQUIV = ["Chaos Control Trap", "Freeze Trap", "Frozen Trap", "Honey Trap", "Paralyze Trap", "Stun Trap", "Bubble Trap"] | # ICE_TRAP_EQUIV = ["Chaos Control Trap", "Freeze Trap", "Frozen Trap", "Honey Trap", "Paralyze Trap", "Stun Trap", "Bubble Trap"] | ||||||
| # DAMAGE_TRAP_EQUIV = ["Banana Trap", "Bomb", "Bonk Trap", "Fire Trap", "Laughter Trap", "Nut Trap", "Push Trap", "Squash Trap", "Thwimp Trap", "TNT Barrel Trap", "Meteor Trap"] | # DAMAGE_TRAP_EQUIV = ["Banana Trap", "Bomb", "Bonk Trap", "Fire Trap", "Laughter Trap", "Nut Trap", "Push Trap", | ||||||
|  | # "Squash Trap", "Thwimp Trap", "TNT Barrel Trap", "Meteor Trap", "Double Damage", "Spike Ball Trap"] | ||||||
|  |  | ||||||
| # SPRING_TRAP_EQUIV = ["Eject Ability", "Hiccup Trap", "Jump Trap", "Jumping Jacks Trap", "Whoops! Trap"] | # SPRING_TRAP_EQUIV = ["Eject Ability", "Hiccup Trap", "Jump Trap", "Jumping Jacks Trap", "Whoops! Trap"] | ||||||
| # HOME_TRAP_EQUIV = ["Blue Balls Curse", "Instant Death Trap"] | # HOME_TRAP_EQUIV = ["Blue Balls Curse", "Instant Death Trap", "Get Out Trap"] | ||||||
| # SLOWNESS_TRAP_EQUIV = ["Iron Boots Trap", "Slow Trap", "Sticky Floor Trap"] | # SLOWNESS_TRAP_EQUIV = ["Iron Boots Trap", "Slow Trap", "Sticky Floor Trap"] | ||||||
| # CUTSCENE_TRAP_EQUIV = ["Phone Trap"] | # CUTSCENE_TRAP_EQUIV = ["Phone Trap"] | ||||||
| # ELEC_TRAP_EQUIV = [] | # ELEC_TRAP_EQUIV = [] | ||||||
|  | # DEPL_TRAP_EQUIV = ["Dry Trap"] | ||||||
|  |  | ||||||
| def grinch_items_to_id() -> dict[str, int]: | def grinch_items_to_id() -> dict[str, int]: | ||||||
|     item_mappings: dict[str, int] = {} |     item_mappings: dict[str, int] = {} | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ from BaseClasses import Location, Region | |||||||
|  |  | ||||||
| class GrinchLocationData(NamedTuple): | class GrinchLocationData(NamedTuple): | ||||||
|     region: str |     region: str | ||||||
|     location_group: str |     location_group: Optional[list[str]] | ||||||
|     id: Optional[int] |     id: Optional[int] | ||||||
|     update_ram_addr: list[GrinchRamData] |     update_ram_addr: list[GrinchRamData] | ||||||
|     reset_addr: Optional[list[GrinchRamData]] = None # Addresses to update once we find the item |     reset_addr: Optional[list[GrinchRamData]] = None # Addresses to update once we find the item | ||||||
| @@ -28,160 +28,178 @@ class GrinchLocation(Location): | |||||||
|         self.type = data.location_group |         self.type = data.location_group | ||||||
|         self.address = self.address |         self.address = self.address | ||||||
|  |  | ||||||
|  | def get_location_names_per_category() -> dict[str, set[str]]: | ||||||
|  |     categories: dict[str, set[str]] = {} | ||||||
|  |  | ||||||
|  |     for name, data in grinch_locations.items(): | ||||||
|  |         if data.location_group is None: | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |         for group in data.location_group:  # iterate over each category | ||||||
|  |             categories.setdefault(group, set()).add(name) | ||||||
|  |  | ||||||
|  |     return categories | ||||||
|  |  | ||||||
| grinch_locations = { | grinch_locations = { | ||||||
| #Going to use current map id as indicator whether or not you visited a location | #Going to use current map id as indicator whether or not you visited a location | ||||||
| #Visitsanity | #Visitsanity | ||||||
|         "Enter Whoville": GrinchLocationData("Whoville", "Visitsanity", 100, [GrinchRamData(0x010000, value=0x07)]), |         "WV - First Visit": GrinchLocationData("Whoville", ["Visitsanity", "Whoville"], 100, [GrinchRamData(0x010000, value=0x07)]), | ||||||
|         "Enter the Post Office": GrinchLocationData("Post Office", "Visitsanity", 101, [GrinchRamData(0x010000, value=0x0A)]), |         "WV - Post Office - First Visit": GrinchLocationData("Post Office", ["Visitsanity", "Whoville", "Post Office"], 101, [GrinchRamData(0x010000, value=0x0A)]), | ||||||
|         "Enter the Town Hall": GrinchLocationData("City Hall", "Visitsanity", 102, [GrinchRamData(0x010000, value=0x08)]), |         "WV - City Hall - First Visit": GrinchLocationData("City Hall", ["Visitsanity", "Whoville", "City Hall"], 102, [GrinchRamData(0x010000, value=0x08)]), | ||||||
|         "Enter the Countdown-To-Xmas Clock Tower": GrinchLocationData("Countdown to X-Mas Clock Tower", "Visitsanity", 103, [GrinchRamData(0x010000, value=0x09)]), |         "WV - Clock Tower - First Visit": GrinchLocationData("Clock Tower", ["Visitsanity", "Whoville", "Clock Tower"], 103, [GrinchRamData(0x010000, value=0x09)]), | ||||||
|         "Enter Who Forest": GrinchLocationData("Who Forest", "Visitsanity", 104, [GrinchRamData(0x010000, value=0x0B)]), |         "WF - First Visit": GrinchLocationData("Who Forest", ["Visitsanity", "Who Forest"], 104, [GrinchRamData(0x010000, value=0x0B)]), | ||||||
|         "Enter the Ski Resort": GrinchLocationData("Ski Resort", "Visitsanity", 105, [GrinchRamData(0x010000, value=0x0C)]), |         "WF - Ski Resort - First Visit": GrinchLocationData("Ski Resort", ["Visitsanity", "Who Forest", "Ski Resort"], 105, [GrinchRamData(0x010000, value=0x0C)]), | ||||||
|         "Enter the Civic Center": GrinchLocationData("Civic Center", "Visitsanity", 106, [GrinchRamData(0x010000, value=0x0D)]), |         "WF - Civic Center - First Visit": GrinchLocationData("Civic Center", ["Visitsanity", "Who Forest", "Civic Center"], 106, [GrinchRamData(0x010000, value=0x0D)]), | ||||||
|         "Enter Who Dump": GrinchLocationData("Who Dump", "Visitsanity", 107, [GrinchRamData(0x010000, value=0x0E)]), |         "WD - First Visit": GrinchLocationData("Who Dump", ["Visitsanity", "Who Dump"], 107, [GrinchRamData(0x010000, value=0x0E)]), | ||||||
|         "Enter the Minefield": GrinchLocationData("Minefield", "Visitsanity", 108, [GrinchRamData(0x010000, value=0x11)]), |         "WD - Minefield - First Visit": GrinchLocationData("Minefield", ["Visitsanity", "Who Dump", "Minefield"], 108, [GrinchRamData(0x010000, value=0x11)]), | ||||||
|         "Enter the Power Plant": GrinchLocationData("Power Plant", "Visitsanity", 109, [GrinchRamData(0x010000, value=0x10)]), |         "WD - Power Plant - First Visit": GrinchLocationData("Power Plant", ["Visitsanity", "Who Dump", "Power Plant"], 109, [GrinchRamData(0x010000, value=0x10)]), | ||||||
|         "Enter the Generator Building": GrinchLocationData("Generator Building", "Visitsanity", 110, [GrinchRamData(0x010000, value=0x0F)]), |         "WD - Generator Building - First Visit": GrinchLocationData("Generator Building", ["Visitsanity", "Who Dump", "Generator Building"], 110, [GrinchRamData(0x010000, value=0x0F)]), | ||||||
|         "Enter Who Lake": GrinchLocationData("Who Lake", "Visitsanity", 111, [GrinchRamData(0x010000, value=0x12)]), |         "WL - South Shore - First Visit": GrinchLocationData("Who Lake", ["Visitsanity", "Who Lake", "South Shore"], 111, [GrinchRamData(0x010000, value=0x12)]), | ||||||
|         "Enter the Submarine World": GrinchLocationData("Submarine World", "Visitsanity", 112, [GrinchRamData(0x010000, value=0x17)]), |         "WL - Submarine World - First Visit": GrinchLocationData("Submarine World", ["Visitsanity", "Who Lake", "Submarine World"], 112, [GrinchRamData(0x010000, value=0x17)]), | ||||||
|         "Enter the Scout's Hut": GrinchLocationData("Scout's Hut", "Visitsanity", 113, [GrinchRamData(0x010000, value=0x13)]), |         "WL - Scout's Hut - First Visit": GrinchLocationData("Scout's Hut", ["Visitsanity", "Who Lake", "Scout's Hut"], 113, [GrinchRamData(0x010000, value=0x13)]), | ||||||
|         "Enter the North Shore": GrinchLocationData("North Shore", "Visitsanity", 114, [GrinchRamData(0x010000, value=0x14)]), |         "WL - North Shore - First Visit": GrinchLocationData("North Shore", ["Visitsanity", "Who Lake", "North Shore"], 114, [GrinchRamData(0x010000, value=0x14)]), | ||||||
|         "Enter the Mayor's Villa": GrinchLocationData("Mayor's Villa", "Visitsanity", 115, [GrinchRamData(0x010000, value=0x16)]), |         "WL - Mayor's Villa - First Visit": GrinchLocationData("Mayor's Villa", ["Visitsanity", "Who Lake", "Mayor's Villa"], 115, [GrinchRamData(0x010000, value=0x16)]), | ||||||
| #Need to find mission completion address for handful of locations that are not documented. | #Need to find mission completion address for handful of locations that are not documented. | ||||||
| #Missions that have value are those ones we need to find the check for | #Missions that have value are those ones we need to find the check for | ||||||
| #Whoville Missions | #Whoville Missions | ||||||
|         "Shuffling The Mail": GrinchLocationData("Post Office", "Whoville Missions", 201, [GrinchRamData(0x0100BE, binary_bit_pos=0)]), |         "WV - Post Office - Shuffling The Mail": GrinchLocationData("Post Office", ["Whoville Missions", "Missions", "Whoville", "Post Office"], 201, [GrinchRamData(0x0100BE, binary_bit_pos=0)]), | ||||||
|         "Smashing Snowmen": GrinchLocationData("Whoville", "Whoville Missions", 200, [GrinchRamData(0x0100C5, value=10)]), |         "WV - Smashing Snowmen": GrinchLocationData("Whoville", ["Whoville Missions", "Missions", "Whoville"], 200, [GrinchRamData(0x0100C5, value=10)]), | ||||||
|         "Painting The Mayor's Posters": GrinchLocationData("Whoville", "Whoville Missions", 202, [GrinchRamData(0x0100C6, value=10)]), |         "WV - Painting The Mayor's Posters": GrinchLocationData("Whoville", ["Whoville Missions", "Missions", "Whoville"], 202, [GrinchRamData(0x0100C6, value=10)]), | ||||||
|         "Launching Eggs Into Houses": GrinchLocationData("Whoville", "Whoville Missions", 203, [GrinchRamData(0x0100C7, value=10)]), |         "WV - Launching Eggs Into Houses": GrinchLocationData("Whoville", ["Whoville Missions", "Missions", "Whoville"], 203, [GrinchRamData(0x0100C7, value=10)]), | ||||||
|         "Modifying The Mayor's Statue": GrinchLocationData("City Hall", "Whoville Missions", 204, [GrinchRamData(0x0100BE, binary_bit_pos=1)]), |         "WV - City Hall - Modifying The Mayor's Statue": GrinchLocationData("City Hall", ["Whoville Missions", "Missions", "Whoville", "City Hall"], 204, [GrinchRamData(0x0100BE, binary_bit_pos=1)]), | ||||||
|         "Advancing The Countdown-To-Xmas Clock": GrinchLocationData("Countdown to X-Mas Clock Tower", "Whoville Missions", 205, [GrinchRamData(0x0100BE, binary_bit_pos=2)]), |         "WV - Clock Tower - Advancing The Countdown-To-Xmas Clock": GrinchLocationData("Clock Tower", ["Whoville Missions", "Missions", "Whoville", "Clock Tower"], 205, [GrinchRamData(0x0100BE, binary_bit_pos=2)]), | ||||||
|         "Squashing All Gifts in Whoville": GrinchLocationData("Whoville", "Whoville Missions", 206, [GrinchRamData(0x01005C, value=500)]), |         "WV - Squashing All Gifts": GrinchLocationData("Whoville", ["Whoville Missions", "Missions", "Giftsanity", "Whoville"], 206, [GrinchRamData(0x01005C, value=500, bit_size=2)]), | ||||||
| #Who Forest Missions | #Who Forest Missions | ||||||
|         "Making Xmas Trees Droop": GrinchLocationData("Who Forest", "Who Forest Missions", 300, [GrinchRamData(0x0100C8, value=10)]), |         "WF - Making Xmas Trees Droop": GrinchLocationData("Who Forest", ["Who Forest Missions", "Missions", "Who Forest"], 300, [GrinchRamData(0x0100C8, value=10)]), | ||||||
|         "Sabotaging Snow Cannon With Glue": GrinchLocationData("Who Forest", "Who Forest Missions", 301, [GrinchRamData(0x0100BE, binary_bit_pos=3)]), |         "WF - Sabotaging Snow Cannon With Glue": GrinchLocationData("Who Forest", ["Who Forest Missions", "Missions", "Who Forest"], 301, [GrinchRamData(0x0100BE, binary_bit_pos=3)]), | ||||||
|         "Putting Beehives In Cabins": GrinchLocationData("Who Forest", "Who Forest Missions", 302, [GrinchRamData(0x0100CA, value=10)]), |         "WF - Putting Beehives In Cabins": GrinchLocationData("Who Forest", ["Who Forest Missions", "Missions", "Who Forest"], 302, [GrinchRamData(0x0100CA, value=10)]), | ||||||
|         "Sliming The Mayor's Skis": GrinchLocationData("Ski Resort", "Who Forest Missions", 303, [GrinchRamData(0x0100BE, binary_bit_pos=4)]), |         "WF - Ski Resort - Sliming The Mayor's Skis": GrinchLocationData("Ski Resort", ["Who Forest Missions", "Missions", "Who Forest", "Ski Resort"], 303, [GrinchRamData(0x0100BE, binary_bit_pos=4)]), | ||||||
|         "Replacing The Candles On The Cake With Fireworks": GrinchLocationData("Civic Center", "Who Forest Missions", 304, [GrinchRamData(0x0100BE, binary_bit_pos=5)]), |         "WF - Civic Center - Replacing The Candles On The Cake With Fireworks": GrinchLocationData("Civic Center", ["Who Forest Missions", "Missions", "Who Forest", "Civic Center"], 304, [GrinchRamData(0x0100BE, binary_bit_pos=5)]), | ||||||
|         "Squashing All Gifts in Who Forest": GrinchLocationData("Who Forest", "Who Forest Missions", 305, [GrinchRamData(0x01005E, value=750)]), |         "WF - Squashing All Gifts": GrinchLocationData("Who Forest", ["Who Forest Missions", "Missions", "Giftsanity", "Who Forest"], 305, [GrinchRamData(0x01005E, value=750, bit_size=2)]), | ||||||
| #Who Dump Missions | #Who Dump Missions | ||||||
|         "Stealing Food From Birds": GrinchLocationData("Who Dump", "Who Dump Missions", 400, [GrinchRamData(0x0100CB, value=10)]), |         "WD - Stealing Food From Birds": GrinchLocationData("Who Dump", ["Who Dump Missions", "Missions", "Who Dump"], 400, [GrinchRamData(0x0100CB, value=10)]), | ||||||
|         "Feeding The Computer With Robot Parts": GrinchLocationData("Who Dump", "Who Dump Missions", 401, [GrinchRamData(0x0100BF, binary_bit_pos=2)]), |         "WD - Feeding The Computer With Robot Parts": GrinchLocationData("Who Dump", ["Who Dump Missions", "Missions", "Who Dump"], 401, [GrinchRamData(0x0100BF, binary_bit_pos=2)]), | ||||||
|         "Infesting The Mayor's House With Rats": GrinchLocationData("Who Dump", "Who Dump Missions", 402, [GrinchRamData(0x0100BE, binary_bit_pos=6)]), |         "WD - Infesting The Mayor's House With Rats": GrinchLocationData("Who Dump", ["Who Dump Missions", "Missions", "Who Dump"], 402, [GrinchRamData(0x0100BE, binary_bit_pos=6)]), | ||||||
|         "Conducting The Stinky Gas To Who-Bris' Shack": GrinchLocationData("Who Dump", "Who Dump Missions", 403, [GrinchRamData(0x0100BE, binary_bit_pos=7)]), |         "WD - Conducting The Stinky Gas To Who-Bris' Shack": GrinchLocationData("Who Dump", ["Who Dump Missions", "Missions", "Who Dump"], 403, [GrinchRamData(0x0100BE, binary_bit_pos=7)]), | ||||||
|         "Shaving Who Dump Guardian": GrinchLocationData("Minefield", "Who Dump Missions", 404, [GrinchRamData(0x0100BF, binary_bit_pos=0)]), |         "WD - Minefield - Shaving Who Dump Guardian": GrinchLocationData("Minefield", ["Who Dump Missions", "Missions", "Who Dump", "Minefield"], 404, [GrinchRamData(0x0100BF, binary_bit_pos=0)]), | ||||||
|         "Short-Circuiting Power-Plant": GrinchLocationData("Generator Building", "Who Dump Missions", 405, [GrinchRamData(0x0100BF, binary_bit_pos=1)]), |         "WD - Generator Building - Short-Circuiting Power-Plant": GrinchLocationData("Generator Building", ["Who Dump Missions", "Missions", "Who Dump", "Generator Building"], 405, [GrinchRamData(0x0100BF, binary_bit_pos=1)]), | ||||||
|         "Squashing All Gifts in Who Dump": GrinchLocationData("Who Dump", "Who Dump Missions", 406, [GrinchRamData(0x010060, value=750)]), |         "WD - Squashing All Gifts": GrinchLocationData("Who Dump", ["Who Dump Missions", "Missions", "Who Dump", "Giftsanity"], 406, [GrinchRamData(0x010060, value=750, bit_size=2)]), | ||||||
| #Who Lake Missions | #Who Lake Missions | ||||||
|         "Putting Thistles In Shorts": GrinchLocationData("Who Lake", "Who Lake Missions", 500, [GrinchRamData(0x0100E5, value=10)]), |         "WL - South Shore - Putting Thistles In Shorts": GrinchLocationData("Who Lake", ["Who Lake Missions", "Missions", "Who Lake", "South Shore", "South Shore Missions"], 500, [GrinchRamData(0x0100E5, value=10)]), | ||||||
|         "Sabotaging The Tents": GrinchLocationData("Who Lake", "Who Lake Missions", 501, [GrinchRamData(0x0100E6, value=10)]), |         "WL - South Shore - Sabotaging The Tents": GrinchLocationData("Who Lake", ["Who Lake Missions", "Missions", "Who Lake", "South Shore", "South Shore Missions"], 501, [GrinchRamData(0x0100E6, value=10)]), | ||||||
|         "Drilling Holes In Canoes": GrinchLocationData("North Shore", "Who Lake Missions", 502, [GrinchRamData(0x0100EE, value=10)]), |         "WL - North Shore - Drilling Holes In Canoes": GrinchLocationData("North Shore", ["Who Lake Missions", "Missions", "Who Lake", "North Shore"], 502, [GrinchRamData(0x0100EE, value=10)]), | ||||||
|         "Modifying The Marine Mobile": GrinchLocationData("Submarine World", "Who Lake Missions", 503, [GrinchRamData(0x0100BF, binary_bit_pos=4)]), |         "WL - Submarine World - Modifying The Marine Mobile": GrinchLocationData("Submarine World", ["Who Lake Missions", "Missions", "Who Lake", "Submarine World"], 503, [GrinchRamData(0x0100BF, binary_bit_pos=4)]), | ||||||
|         "Hooking The Mayor's Bed To The Motorboat": GrinchLocationData("Mayor's Villa", "Who Lake Missions", 504, [GrinchRamData(0x0100BF, binary_bit_pos=3)]), |         "WL - Mayor's Villa - Hooking The Mayor's Bed To The Motorboat": GrinchLocationData("Mayor's Villa", ["Who Lake Missions", "Missions", "Who Lake", "Mayor's Villa"], 504, [GrinchRamData(0x0100BF, binary_bit_pos=3)]), | ||||||
|         "Squashing All Gifts in Who Lake": GrinchLocationData("Who Lake", "Who Lake Missions", 505, [GrinchRamData(0x010062, value=1000)]), |         "WL - Squashing All Gifts": GrinchLocationData("Who Lake", ["Who Lake Missions", "Missions", "Who Lake", "Giftsanity"], 505, [GrinchRamData(0x010062, value=1000, bit_size=2)]), | ||||||
| #Need to find binary values for individual blueprints, but all ram addresses are found | #Need to find binary values for individual blueprints, but all ram addresses are found | ||||||
| #Blueprints | #Blueprints | ||||||
| #Binoculars Blueprints | #Binoculars Blueprints | ||||||
|         "Binoculars Blueprint - Post Office Roof": GrinchLocationData("Whoville", "Binocular Blueprints", 600, [GrinchRamData(0x01020B, binary_bit_pos=2)]), |         "WV - Binoculars BP on Post Office Roof": GrinchLocationData("Whoville", ["Binocular Blueprints", "Blueprints", "Whoville", "Whoville Blueprints"], 600, [GrinchRamData(0x01020B, binary_bit_pos=2)]), | ||||||
|         "Binoculars Blueprint - City Hall Library - Left Side": GrinchLocationData("City Hall", "Binocular Blueprints", 601, [GrinchRamData(0x01021F, binary_bit_pos=6)]), |         "WV - City Hall - Binoculars BP left side of Library": GrinchLocationData("City Hall", ["Binocular Blueprints", "Blueprints", "Whoville", "Whoville Blueprints", "City Hall", "City Hall Blueprints"], 601, [GrinchRamData(0x01021F, binary_bit_pos=6)]), | ||||||
|         "Binoculars Blueprint - City Hall Library - Front Side": GrinchLocationData("City Hall", "Binocular Blueprints", 602, [GrinchRamData(0x01021F, binary_bit_pos=5)]), |         "WV - City Hall - Binoculars BP front side of Library": GrinchLocationData("City Hall", ["Binocular Blueprints", "Blueprints", "Whoville", "Whoville Blueprints", "City Hall", "City Hall Blueprints"], 602, [GrinchRamData(0x01021F, binary_bit_pos=5)]), | ||||||
|         "Binoculars Blueprint - City Hall Library - Right Side": GrinchLocationData("City Hall", "Binocular Blueprints", 603, [GrinchRamData(0x01021F, binary_bit_pos=4)]), |         "WV - City Hall - Binoculars BP right side of Library": GrinchLocationData("City Hall", ["Binocular Blueprints", "Blueprints", "Whoville", "Whoville Blueprints", "City Hall", "City Hall Blueprints"], 603, [GrinchRamData(0x01021F, binary_bit_pos=4)]), | ||||||
| #Rotten Egg Launcher Blueprints | #Rotten Egg Launcher Blueprints | ||||||
|         "REL Blueprint - Outside City Hall": GrinchLocationData("Whoville", "Rotten Egg Launcher Blueprints", 700, [GrinchRamData(0x01020B, binary_bit_pos=0)]), |         "WV - REL BP left of City Hall": GrinchLocationData("Whoville", ["Rotten Egg Launcher Blueprints", "Blueprints", "Whoville", "Whoville Blueprints"], 700, [GrinchRamData(0x01020B, binary_bit_pos=0)]), | ||||||
|         "REL Blueprint - Outside Clock Tower": GrinchLocationData("Whoville", "Rotten Egg Launcher Blueprints", 701, [GrinchRamData(0x01020B, binary_bit_pos=1)]), |         "WV - REL BP left of Clock Tower": GrinchLocationData("Whoville", ["Rotten Egg Launcher Blueprints", "Blueprints", "Whoville", "Whoville Blueprints"], 701, [GrinchRamData(0x01020B, binary_bit_pos=1)]), | ||||||
|         "REL Blueprint - Post Office - Inside Silver Room": GrinchLocationData("Post Office", "Rotten Egg Launcher Blueprints", 702, [GrinchRamData(0x01021C, binary_bit_pos=1)]), |         "WV - Post Office - REL BP inside Silver Room": GrinchLocationData("Post Office", ["Rotten Egg Launcher Blueprints", "Blueprints", "Whoville", "Whoville Blueprints", "Post Office", "Post Office Blueprints"], 702, [GrinchRamData(0x01021C, binary_bit_pos=1)]), | ||||||
|         "REL Blueprint - Post Office - After Mission Completion": GrinchLocationData("Post Office", "Rotten Egg Launcher Blueprints", 703, [GrinchRamData(0x01021C, binary_bit_pos=2)]), |         "WV - Post Office - REL BP at Entrance Door after Mission Completion": GrinchLocationData("Post Office", ["Rotten Egg Launcher Blueprints", "Blueprints", "Whoville", "Whoville Blueprints", "Post Office", "Post Office Blueprints"], 703, [GrinchRamData(0x01021C, binary_bit_pos=2)]), | ||||||
| #Rocket Spring Blueprints | #Rocket Spring Blueprints | ||||||
|         "RS Blueprint - Behind Vacuum": GrinchLocationData("Who Forest", "Rocket Spring Blueprints", 800, [GrinchRamData(0x010243, binary_bit_pos=3)]), |         "WF - RS BP behind Vacuum Tube": GrinchLocationData("Who Forest", ["Rocket Spring Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 800, [GrinchRamData(0x010243, binary_bit_pos=3)]), | ||||||
|         "RS Blueprint - Front of 2nd House near entrance": GrinchLocationData("Who Forest", "Rocket Spring Blueprints", 801, [GrinchRamData(0x010243, binary_bit_pos=1)]), |         "WF - RS BP in front of 2nd House near Vacuum Tube": GrinchLocationData("Who Forest", ["Rocket Spring Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 801, [GrinchRamData(0x010243, binary_bit_pos=1)]), | ||||||
|         "RS Blueprint - Near Tree House on Ground": GrinchLocationData("Who Forest", "Rocket Spring Blueprints", 802, [GrinchRamData(0x010243, binary_bit_pos=4)]), |         "WF - RS BP near Tree House on Ground": GrinchLocationData("Who Forest", ["Rocket Spring Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 802, [GrinchRamData(0x010243, binary_bit_pos=4)]), | ||||||
|         "RS Blueprint - Near Cable Car House": GrinchLocationData("Who Forest", "Rocket Spring Blueprints", 804, [GrinchRamData(0x010242, binary_bit_pos=7)]), |         "WF - RS BP behind Cable Car House": GrinchLocationData("Who Forest", ["Rocket Spring Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 804, [GrinchRamData(0x010242, binary_bit_pos=7)]), | ||||||
|         "RS Blueprint - Near Who Snowball in Cave": GrinchLocationData("Who Forest", "Rocket Spring Blueprints", 805, [GrinchRamData(0x010242, binary_bit_pos=6)]), |         "WF - RS BP near Who Snowball in Cave": GrinchLocationData("Who Forest", ["Rocket Spring Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 805, [GrinchRamData(0x010242, binary_bit_pos=6)]), | ||||||
|         "RS Blueprint - Branch Platform Closest to Glue Cannon": GrinchLocationData("Who Forest", "Rocket Spring Blueprints", 806, [GrinchRamData(0x010243, binary_bit_pos=2)]), |         "WF - RS BP on Branch Platform closest to Glue Cannon": GrinchLocationData("Who Forest", ["Rocket Spring Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 806, [GrinchRamData(0x010243, binary_bit_pos=2)]), | ||||||
|         "RS Blueprint - Branch Platform Near Beast": GrinchLocationData("Who Forest", "Rocket Spring Blueprints", 807, [GrinchRamData(0x010243, binary_bit_pos=0)]), |         "WF - RS BP on Branch Platform Near Beast": GrinchLocationData("Who Forest", ["Rocket Spring Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 807, [GrinchRamData(0x010243, binary_bit_pos=0)]), | ||||||
|         "RS Blueprint - Branch Platform Ledge Grab House": GrinchLocationData("Who Forest", "Rocket Spring Blueprints", 808, [GrinchRamData(0x010243, binary_bit_pos=6)]), |         "WF - RS BP on Branch Platform Elevated next to House": GrinchLocationData("Who Forest", ["Rocket Spring Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 808, [GrinchRamData(0x010243, binary_bit_pos=6)]), | ||||||
|         "RS Blueprint - On Tree House": GrinchLocationData("Who Forest", "Rocket Spring Blueprints", 809, [GrinchRamData(0x010243, binary_bit_pos=5)]), |         "WF - RS BP on Tree House": GrinchLocationData("Who Forest", ["Rocket Spring Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 809, [GrinchRamData(0x010243, binary_bit_pos=5)]), | ||||||
| #Slime Shooter Blueprints | #Slime Shooter Blueprints | ||||||
|         "SS Blueprint - Branch Platform Elevated House": GrinchLocationData("Who Forest", "Slime Shooter Blueprints", 900, [GrinchRamData(0x010244, binary_bit_pos=3)]), |         "WF - SS BP in Branch Platform Elevated House": GrinchLocationData("Who Forest", ["Slime Shooter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 900, [GrinchRamData(0x010244, binary_bit_pos=3)]), | ||||||
|         "SS Blueprint - Branch Platform House next to Beast": GrinchLocationData("Who Forest", "Slime Shooter Blueprint", 901, [GrinchRamData(0x010243, binary_bit_pos=7)]), |         "WF - SS BP in Branch Platform House next to Beast": GrinchLocationData("Who Forest", ["Slime Shooter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 901, [GrinchRamData(0x010243, binary_bit_pos=7)]), | ||||||
|         "SS Blueprint - House near Civic Center Cave": GrinchLocationData("Who Forest", "Slime Shooter Blueprints", 902, [GrinchRamData(0x010244, binary_bit_pos=2)]), |         "WF - SS BP in House in front of Civic Center Cave": GrinchLocationData("Who Forest", ["Slime Shooter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 902, [GrinchRamData(0x010244, binary_bit_pos=2)]), | ||||||
|         "SS Blueprint - House next to Tree House": GrinchLocationData("Who Forest", "Slime Shooter Blueprints", 903, [GrinchRamData(0x010244, binary_bit_pos=1)]), |         "WF - SS BP in House next to Tree House": GrinchLocationData("Who Forest", ["Slime Shooter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 903, [GrinchRamData(0x010244, binary_bit_pos=1)]), | ||||||
|         "SS Blueprint - House across from Tree House": GrinchLocationData("Who Forest", "Slime Shooter Blueprints", 904, [GrinchRamData(0x010244, binary_bit_pos=5)]), |         "WF - SS BP in House across from Tree House": GrinchLocationData("Who Forest", ["Slime Shooter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 904, [GrinchRamData(0x010244, binary_bit_pos=5)]), | ||||||
|         "SS Blueprint - 2nd House near entrance right side": GrinchLocationData("Who Forest", "Slime Shooter Blueprints", 905, [GrinchRamData(0x010244, binary_bit_pos=4)]), |         "WF - SS BP in 2nd House near Vacuum Tube Right Side": GrinchLocationData("Who Forest", ["Slime Shooter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 905, [GrinchRamData(0x010244, binary_bit_pos=4)]), | ||||||
|         "SS Blueprint - 2nd House near entrance left side": GrinchLocationData("Who Forest", "Slime Shooter Blueprints", 906, [GrinchRamData(0x010244, binary_bit_pos=7)]), |         "WF - SS BP in 2nd House near Vacuum Tube Left Side": GrinchLocationData("Who Forest", ["Slime Shooter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 906, [GrinchRamData(0x010244, binary_bit_pos=7)]), | ||||||
|         "SS Blueprint - 2nd House near entrance inbetween blueprints": GrinchLocationData("Who Forest", "Slime Shooter Blueprints", 907, [GrinchRamData(0x010244, binary_bit_pos=6)]), |         "WF - SS BP in 2nd House near Vacuum Tube inbetween Blueprints": GrinchLocationData("Who Forest", ["Slime Shooter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 907, [GrinchRamData(0x010244, binary_bit_pos=6)]), | ||||||
|         "SS Blueprint - House near entrance": GrinchLocationData("Who Forest", "Slime Shooter Blueprints", 908, [GrinchRamData(0x010244, binary_bit_pos=0)]), |         "WF - SS BP in House near Vacuum Tube": GrinchLocationData("Who Forest", ["Slime Shooter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints"], 908, [GrinchRamData(0x010244, binary_bit_pos=0)]), | ||||||
| #Octopus Climbing Device | #Octopus Climbing Device | ||||||
|         "OCD Blueprint - Middle Pipe": GrinchLocationData("Who Dump", "Octopus Climbing Device Blueprints", 1001, [GrinchRamData(0x010252, binary_bit_pos=3)]), |         "WD - OCD BP inside Middle Pipe": GrinchLocationData("Who Dump", ["Octopus Climbing Device Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints"], 1001, [GrinchRamData(0x010252, binary_bit_pos=3)]), | ||||||
|         "OCD Blueprint - Right Pipe": GrinchLocationData("Who Dump", "Octopus Climbing Device Blueprints", 1002, [GrinchRamData(0x010252, binary_bit_pos=5)]), |         "WD - OCD BP inside Right Pipe": GrinchLocationData("Who Dump", ["Octopus Climbing Device Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints"], 1002, [GrinchRamData(0x010252, binary_bit_pos=5)]), | ||||||
|         "OCD Blueprint - Mayor's House Rat Vent": GrinchLocationData("Who Dump", "Octopus Climbing Device Blueprints", 1003, [GrinchRamData(0x010252, binary_bit_pos=1)]), |         "WD - OCD BP in Vent to Mayor's House": GrinchLocationData("Who Dump", ["Octopus Climbing Device Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints"], 1003, [GrinchRamData(0x010252, binary_bit_pos=1)]), | ||||||
|         "OCD Blueprint - Left Pipe": GrinchLocationData("Who Dump", "Octopus Climbing Device Blueprints", 1004, [GrinchRamData(0x010252, binary_bit_pos=4)]), |         "WD - OCD BP inside Left Pipe": GrinchLocationData("Who Dump", ["Octopus Climbing Device Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints"], 1004, [GrinchRamData(0x010252, binary_bit_pos=4)]), | ||||||
|         "OCD Blueprint - Near Power Plant Wall on right side": GrinchLocationData("Who Dump", "Octopus Climbing Device Blueprints", 1005, [GrinchRamData(0x010252, binary_bit_pos=0)]), |         "WD - OCD BP near Right Side of Power Plant Wall": GrinchLocationData("Who Dump", ["Octopus Climbing Device Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints"], 1005, [GrinchRamData(0x010252, binary_bit_pos=0)]), | ||||||
|         "OCD Blueprint - Near Who-Bris' Shack": GrinchLocationData("Who Dump", "Octopus Climbing Device Blueprints", 1006, [GrinchRamData(0x010252, binary_bit_pos=2)]), |         "WD - OCD BP near Who-Bris' Shack": GrinchLocationData("Who Dump", ["Octopus Climbing Device Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints"], 1006, [GrinchRamData(0x010252, binary_bit_pos=2)]), | ||||||
|         "OCD Blueprint - Guardian's House - Left Side": GrinchLocationData("Minefield", "Octopus Climbing Device Blueprints", 1007, [GrinchRamData(0x01026E, binary_bit_pos=2)]), |         "WD - Minefield - OCD BP on Left Side of House": GrinchLocationData("Minefield", ["Octopus Climbing Device Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints", "Minefield", "Minefield Blueprints"], 1007, [GrinchRamData(0x01026E, binary_bit_pos=2)]), | ||||||
|         "OCD Blueprint - Guardian's House - Right Side": GrinchLocationData("Minefield", "Octopus Climbing Device Blueprints", 1008, [GrinchRamData(0x01026E, binary_bit_pos=4)]), |         "WD - Minefield - OCD BP on Right Side of Shack": GrinchLocationData("Minefield", ["Octopus Climbing Device Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints", "Minefield", "Minefield Blueprints"], 1008, [GrinchRamData(0x01026E, binary_bit_pos=4)]), | ||||||
|         "OCD Blueprint - Inside Guardian's House": GrinchLocationData("Minefield", "Octopus Climbing Device Blueprints", 1009, [GrinchRamData(0x01026E, binary_bit_pos=3)]), |         "WD - Minefield - OCD BP inside Guardian's House": GrinchLocationData("Minefield", ["Octopus Climbing Device Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints", "Minefield", "Minefield Blueprints"], 1009, [GrinchRamData(0x01026E, binary_bit_pos=3)]), | ||||||
| #Marine Mobile Blueprints | #Marine Mobile Blueprints | ||||||
|         "MM Blueprint - South Shore - Bridge to Scout's Hut": GrinchLocationData("Who Lake", "Marine Mobile Blueprints", 1100, [GrinchRamData(0x010281, binary_bit_pos=5)]), |         "WL - South Shore - MM BP on Bridge to Scout's Hut": GrinchLocationData("Who Lake", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "South Shore", "South Shore Blueprints"], 1100, [GrinchRamData(0x010281, binary_bit_pos=5)]), | ||||||
|         "MM Blueprint - South Shore - Tent near Porcupine": GrinchLocationData("Who Lake", "Marine Mobile Blueprints", 1101, [GrinchRamData(0x010281, binary_bit_pos=6)]), |         "WL - South Shore - MM BP across from Tent near Porcupine": GrinchLocationData("Who Lake", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "South Shore", "South Shore Blueprints"], 1101, [GrinchRamData(0x010281, binary_bit_pos=6)]), | ||||||
|         "MM Blueprint - South Shore - Near Outhouse": GrinchLocationData("Who Lake", "Marine Mobile Blueprints", 1102, [GrinchRamData(0x010281, binary_bit_pos=7)]), |         "WL - South Shore - MM BP near Outhouse": GrinchLocationData("Who Lake", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "South Shore", "South Shore Blueprints"], 1102, [GrinchRamData(0x010281, binary_bit_pos=7)]), | ||||||
|         "MM Blueprint - South Shore - Near Hill Bridge": GrinchLocationData("Who Lake", "Marine Mobile Blueprints", 1103, [GrinchRamData(0x010282, binary_bit_pos=0)]), |         "WL - South Shore - MM BP near Hill Bridge": GrinchLocationData("Who Lake", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "South Shore", "South Shore Blueprints"], 1103, [GrinchRamData(0x010282, binary_bit_pos=0)]), | ||||||
|         "MM Blueprint - South Shore - Scout's Hut Roof": GrinchLocationData("Who Lake", "Marine Mobile Blueprints", 1104, [GrinchRamData(0x010281, binary_bit_pos=4)]), |         "WL - South Shore - MM BP on Scout's Hut Roof": GrinchLocationData("Who Lake", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "South Shore", "South Shore Blueprints"], 1104, [GrinchRamData(0x010281, binary_bit_pos=4)]), | ||||||
|         "MM Blueprint - South Shore - Grass Platform": GrinchLocationData("Who Lake", "Marine Mobile Blueprints", 1105, [GrinchRamData(0x010281, binary_bit_pos=2)]), |         "WL - South Shore - MM BP on Grass Platform": GrinchLocationData("Who Lake", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "South Shore", "South Shore Blueprints"], 1105, [GrinchRamData(0x010281, binary_bit_pos=2)]), | ||||||
|         "MM Blueprint - South Shore - Zipline by Beast": GrinchLocationData("Who Lake", "Marine Mobile Blueprints", 1106, [GrinchRamData(0x010281, binary_bit_pos=3)]), |         "WL - South Shore - MM BP across Zipline Platform": GrinchLocationData("Who Lake", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "South Shore", "South Shore Blueprints"], 1106, [GrinchRamData(0x010281, binary_bit_pos=3)]), | ||||||
|         "MM Blueprint - South Shore - Behind Summer Beast": GrinchLocationData("Who Lake", "Marine Mobile Blueprints", 1107, [GrinchRamData(0x010282, binary_bit_pos=1)]), |         "WL - South Shore - MM BP behind Summer Beast": GrinchLocationData("Who Lake", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "South Shore", "South Shore Blueprints"], 1107, [GrinchRamData(0x010282, binary_bit_pos=1)]), | ||||||
|         "MM Blueprint - North Shore - Below Bridge": GrinchLocationData("North Shore", "Marine Mobile Blueprints", 1108, [GrinchRamData(0x010293, binary_bit_pos=0)]), |         "WL - North Shore - MM BP below Bridge": GrinchLocationData("North Shore", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "North Shore", "North Shore Blueprints"], 1108, [GrinchRamData(0x010293, binary_bit_pos=0)]), | ||||||
|         "MM Blueprint - North Shore - Behind Skunk Hut": GrinchLocationData("North Shore", "Marine Mobile Blueprints", 1109, [GrinchRamData(0x010293, binary_bit_pos=2)]), |         "WL - North Shore - MM BP behind Skunk Hut": GrinchLocationData("North Shore", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "North Shore", "North Shore Blueprints"], 1109, [GrinchRamData(0x010293, binary_bit_pos=2)]), | ||||||
|         "MM Blueprint - North Shore - Inside Skunk Hut": GrinchLocationData("North Shore", "Marine Mobile Blueprints", 1110, [GrinchRamData(0x010292, binary_bit_pos=6)]), |         "WL - North Shore - MM BP inside Skunk Hut": GrinchLocationData("North Shore", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "North Shore", "North Shore Blueprints"], 1110, [GrinchRamData(0x010292, binary_bit_pos=6)]), | ||||||
|         "MM Blueprint - North Shore - Fenced in Area": GrinchLocationData("North Shore", "Marine Mobile Blueprints", 1111, [GrinchRamData(0x010292, binary_bit_pos=7)]), |         "WL - North Shore - MM BP inside House's Fence": GrinchLocationData("North Shore", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "North Shore", "North Shore Blueprints"], 1111, [GrinchRamData(0x010292, binary_bit_pos=7)]), | ||||||
|         "MM Blueprint - North Shore - Boulder Box near Bridge": GrinchLocationData("North Shore", "Marine Mobile Blueprints", 1112, [GrinchRamData(0x010293, binary_bit_pos=3)]), |         "WL - North Shore - MM BP inside Boulder Box near Bridge": GrinchLocationData("North Shore", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "North Shore", "North Shore Blueprints"], 1112, [GrinchRamData(0x010293, binary_bit_pos=3)]), | ||||||
|         "MM Blueprint - North Shore - Boulder Box behind Skunk Hut": GrinchLocationData("North Shore", "Marine Mobile Blueprints", 1113, [GrinchRamData(0x010293, binary_bit_pos=4)]), |         "WL - North Shore - MM BP inside Boulder Box behind Skunk Hut": GrinchLocationData("North Shore", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "North Shore", "North Shore Blueprints"], 1113, [GrinchRamData(0x010293, binary_bit_pos=4)]), | ||||||
|         "MM Blueprint - North Shore - Inside Drill House": GrinchLocationData("North Shore", "Marine Mobile Blueprints", 1114, [GrinchRamData(0x010292, binary_bit_pos=5)]), |         "WL - North Shore - MM BP inside Drill House": GrinchLocationData("North Shore", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "North Shore", "North Shore Blueprints"], 1114, [GrinchRamData(0x010292, binary_bit_pos=5)]), | ||||||
|         "MM Blueprint - North Shore - Crow Platform near Drill House": GrinchLocationData("North Shore", "Marine Mobile Blueprints", 1115, [GrinchRamData(0x010293, binary_bit_pos=1)]), |         "WL - North Shore - MM BP on Crow Platform near Drill House": GrinchLocationData("North Shore", ["Marine Mobile Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "North Shore", "North Shore Blueprints"], 1115, [GrinchRamData(0x010293, binary_bit_pos=1)]), | ||||||
|         #Grinch Copter Blueprints |         #Grinch Copter Blueprints | ||||||
|         "GC Blueprint - Whoville City Hall - Safe Room": GrinchLocationData("City Hall", "Grinch Copter Blueprints", 1200, [GrinchRamData(0x01021F, binary_bit_pos=7)]), |         "WV - City Hall - GC BP in Safe Room": GrinchLocationData("City Hall", ["Grinch Copter Blueprints", "Blueprints", "Whoville", "Whoville Blueprints", "City Hall", "City Hall Blueprints"], 1200, [GrinchRamData(0x01021F, binary_bit_pos=7)]), | ||||||
|         "GC Blueprint - Whoville City Hall - Statue Room": GrinchLocationData("City Hall", "Grinch Copter Blueprints", 1201, [GrinchRamData(0x010220, binary_bit_pos=0)]), |         "WV - City Hall - GC BP in Statue Room": GrinchLocationData("City Hall", ["Grinch Copter Blueprints", "Blueprints", "Whoville", "Whoville Blueprints", "City Hall", "City Hall Blueprints"], 1201, [GrinchRamData(0x010220, binary_bit_pos=0)]), | ||||||
|         "GC Blueprint - Whoville Clock Tower - Before Bells": GrinchLocationData("Countdown to X-Mas Clock Tower", "Grinch Copter Blueprints", 1202, [GrinchRamData(0x010216, binary_bit_pos=3)]), |         "WV - Clock Tower - GC BP in Bedroom": GrinchLocationData("Clock Tower", ["Grinch Copter Blueprints", "Blueprints", "Whoville", "Whoville Blueprints", "Clock Tower", "Clock Tower Blueprints"], 1202, [GrinchRamData(0x010216, binary_bit_pos=3)]), | ||||||
|         "GC Blueprint - Whoville Clock Tower - After Bells": GrinchLocationData("Countdown to X-Mas Clock Tower", "Grinch Copter Blueprints", 1203, [GrinchRamData(0x010216, binary_bit_pos=2)]), |         "WV - Clock Tower - GC BP in Bell Room": GrinchLocationData("Clock Tower", ["Grinch Copter Blueprints", "Blueprints", "Whoville", "Whoville Blueprints", "Clock Tower", "Clock Tower Blueprints"], 1203, [GrinchRamData(0x010216, binary_bit_pos=2)]), | ||||||
|         "GC Blueprint - Who Forest Ski Resort - Inside Dog's Fence": GrinchLocationData("Ski Resort", "Grinch Copter Blueprints", 1204, [GrinchRamData(0x010234, binary_bit_pos=7)]), |         "WF - Ski Resort - GC BP inside Dog's Fence": GrinchLocationData("Ski Resort", ["Grinch Copter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints", "Ski Resort", "Ski Resort Blueprints"], 1204, [GrinchRamData(0x010234, binary_bit_pos=7)]), | ||||||
|         "GC Blueprint - Who Forest Ski Resort - Max Cave": GrinchLocationData("Ski Resort", "Grinch Copter Blueprints", 1205, [GrinchRamData(0x010234, binary_bit_pos=6)]), |         "WF - Ski Resort - GC BP in Max Cave": GrinchLocationData("Ski Resort", ["Grinch Copter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints", "Ski Resort", "Ski Resort Blueprints"], 1205, [GrinchRamData(0x010234, binary_bit_pos=6)]), | ||||||
|         "GC Blueprint - Who Forest Civic Center - Climb across Bat Cave wall": GrinchLocationData("Civic Center", "Grinch Copter Blueprints", 1206, [GrinchRamData(0x01022A, binary_bit_pos=7)]), |         "WF - Civic Center - GC BP on Left Side in Bat Cave Wall": GrinchLocationData("Civic Center", ["Grinch Copter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints", "Civic Center", "Civic Center Blueprints"], 1206, [GrinchRamData(0x01022A, binary_bit_pos=7)]), | ||||||
|         "GC Blueprint - Who Forest Civic Center - Shoot Icicle in Bat Entrance": GrinchLocationData("Civic Center", "Grinch Copter Blueprints", 1207, [GrinchRamData(0x01022B, binary_bit_pos=0)]), |         "WF - Civic Center - GC BP in Frozen Ice": GrinchLocationData("Civic Center", ["Grinch Copter Blueprints", "Blueprints", "Who Forest", "Who Forest Blueprints", "Civic Center", "Civic Center Blueprints"], 1207, [GrinchRamData(0x01022B, binary_bit_pos=0)]), | ||||||
|         "GC Blueprint - Who Dump Power Plant - Max Cave": GrinchLocationData("Power Plant", "Grinch Copter Blueprints", 1208, [GrinchRamData(0x010265, binary_bit_pos=1)]), |         "WD - Power Plant - GC BP in Max Cave": GrinchLocationData("Power Plant", ["Grinch Copter Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints", "Power Plant", "Power Plant Blueprints"], 1208, [GrinchRamData(0x010265, binary_bit_pos=1)]), | ||||||
|         "GC Blueprint - Who Dump Power Plant - After First Gate": GrinchLocationData("Power Plant", "Grinch Copter Blueprints", 1209, [GrinchRamData(0x010265, binary_bit_pos=2)]), |         "WD - Power Plant - GC BP After First Gate": GrinchLocationData("Power Plant", ["Grinch Copter Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints", "Power Plant", "Power Plant Blueprints"], 1209, [GrinchRamData(0x010265, binary_bit_pos=2)]), | ||||||
|         "GC Blueprint - Who Dump Generator Building - Before Mission": GrinchLocationData("Generator Building", "Grinch Copter Blueprints", 1210, [GrinchRamData(0x01026B, binary_bit_pos=0)]), |         "WD - Generator Building - GC BP on the Highest Platform": GrinchLocationData("Generator Building", ["Grinch Copter Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints", "Generator Building", "Generator Building Blueprints"], 1210, [GrinchRamData(0x01026B, binary_bit_pos=0)]), | ||||||
|         "GC Blueprint - Who Dump Generator Building - After Mission": GrinchLocationData("Generator Building", "Grinch Copter Blueprints", 1211, [GrinchRamData(0x01026B, binary_bit_pos=1)]), |         "WD - Generator Building - GC BP at the Entrance after Mission Completion": GrinchLocationData("Generator Building", ["Grinch Copter Blueprints", "Blueprints", "Who Dump", "Who Dump Blueprints", "Generator Building", "Generator Building Blueprints"], 1211, [GrinchRamData(0x01026B, binary_bit_pos=1)]), | ||||||
|         "GC Blueprint - Who Lake South Shore - Submarine World - Above Surface": GrinchLocationData("Submarine World", "Grinch Copter Blueprints", 1212, [GrinchRamData(0x010289, binary_bit_pos=3)]), |         "WL - Submarine World - GC BP Just Below Water Surface": GrinchLocationData("Submarine World", ["Grinch Copter Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "Submarine World", "Submarine World Blueprints"], 1212, [GrinchRamData(0x010289, binary_bit_pos=3)]), | ||||||
|         "GC Blueprint - Who Lake South Shore - Submarine World - Underwater": GrinchLocationData("Submarine World", "Grinch Copter Blueprints", 1213, [GrinchRamData(0x010289, binary_bit_pos=4)]), |         "WL - Submarine World - GC BP Underwater": GrinchLocationData("Submarine World", ["Grinch Copter Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "Submarine World", "Submarine World Blueprints"], 1213, [GrinchRamData(0x010289, binary_bit_pos=4)]), | ||||||
|         "GC Blueprint - Who Lake North Shore - Mayor's Villa - Tree Branch": GrinchLocationData("Mayor's Villa", "Grinch Copter Blueprints", 1214, [GrinchRamData(0x010275, binary_bit_pos=7)]), |         "WL - Mayor's Villa - GC BP on Tree Branch": GrinchLocationData("Mayor's Villa", ["Grinch Copter Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "Mayor's Villa", "Mayor's Villa Blueprints"], 1214, [GrinchRamData(0x010275, binary_bit_pos=7)]), | ||||||
|         "GC Blueprint - Who Lake North Shore - Mayor's Villa - Cave": GrinchLocationData("Mayor's Villa", "Grinch Copter Blueprints", 1215, [GrinchRamData(0x010275, binary_bit_pos=6)]), |         "WL - Mayor's Villa - GC BP in Pirate's Cave": GrinchLocationData("Mayor's Villa", ["Grinch Copter Blueprints", "Blueprints", "Who Lake", "Who Lake Blueprints", "Mayor's Villa", "Mayor's Villa Blueprints"], 1215, [GrinchRamData(0x010275, binary_bit_pos=6)]), | ||||||
| #Sleigh Room Locations | #Sleigh Room Locations | ||||||
|         "Stealing All Gifts": GrinchLocationData("Sleigh Room", "Sleigh Ride", 1300, [GrinchRamData(0x0100BF, binary_bit_pos=6)]), |         "MC - Sleigh Ride - Stealing All Gifts": GrinchLocationData("Sleigh Room", ["Sleigh Ride"], 1300, [GrinchRamData(0x0100BF, binary_bit_pos=6)]), | ||||||
|         "Neutralizing Santa": GrinchLocationData("Sleigh Room", "Sleigh Ride", None, [GrinchRamData(0x0100BF, binary_bit_pos=7)]), |         "MC - Sleigh Ride - Neutralizing Santa": GrinchLocationData("Sleigh Room", None, None, [GrinchRamData(0x010000, value=0x3E)]),#[GrinchRamData(0x0100BF, binary_bit_pos=7)]), | ||||||
| #Heart of Stones | #Heart of Stones | ||||||
|         "Heart of Stone - Whoville's Post Office": GrinchLocationData("Post Office", "Heart of Stones", 1400, [GrinchRamData(0x0101FA, binary_bit_pos=6)]), |         "WV - Post Office - Heart of Stone": GrinchLocationData("Post Office", ["Heart of Stones", "Whoville", "Post Office"], 1400, [GrinchRamData(0x0101FA, binary_bit_pos=6)]), | ||||||
|         "Heart of Stone - Who Forest's Ski Resort": GrinchLocationData("Ski Resort", "Heart of Stones", 1401, [GrinchRamData(0x0101FA, binary_bit_pos=7)]), |         "WF - Ski Resort - Heart of Stone": GrinchLocationData("Ski Resort", ["Heart of Stones", "Who Forest", "Ski Resort"], 1401, [GrinchRamData(0x0101FA, binary_bit_pos=7)]), | ||||||
|         "Heart of Stone - Who Dump's Minefield": GrinchLocationData("Minefield", "Heart of Stones", 1402, [GrinchRamData(0x0101FB, binary_bit_pos=0)]), |         "WD - Minefield - Heart of Stone": GrinchLocationData("Minefield", ["Heart of Stones", "Who Dump", "Minefield"], 1402, [GrinchRamData(0x0101FB, binary_bit_pos=0)]), | ||||||
|         "Heart of Stone - Who Lake's North Shore": GrinchLocationData("North Shore", "Heart of Stones", 1403, [GrinchRamData(0x0101FB, binary_bit_pos=1)]), |         "WL - North Shore - Heart of Stone": GrinchLocationData("North Shore", ["Heart of Stones", "Who Lake", "North Shore"], 1403, [GrinchRamData(0x0101FB, binary_bit_pos=1)]), | ||||||
| #Supadow Minigames | #Supadow Minigames | ||||||
|         # "Spin N' Win - Easy": GrinchLocationData("Spin N' Win Supadow", "Supadow Minigames", 1500, [GrinchRamData()]), |         # "Spin N' Win - Easy": GrinchLocationData("Spin N' Win", ["Supadow Minigames", "Spin N' Win"], 1500, [GrinchRamData()]), | ||||||
|         # "Spin N' Win - Hard": GrinchLocationData("Spin N' Win Supadow", "Supadow Minigames", 1501, [GrinchRamData()]), |         # "Spin N' Win - Hard": GrinchLocationData("Spin N' Win", ["Supadow Minigames", "Spin N' Win"], 1501, [GrinchRamData()]), | ||||||
|         # "Spin N' Win - Real Tough": GrinchLocationData("Spin N' Win Supadow", "Supadow Minigames", 1502, [GrinchRamData()]), |         # "Spin N' Win - Real Tough": GrinchLocationData("Spin N' Win", ["Supadow Minigames", "Spin N' Win"], 1502, [GrinchRamData()]), | ||||||
|         # "Dankamania - Easy - 15 Points": GrinchLocationData("Dankamania Supadow", "Supadow Minigames", 1503, [GrinchRamData()]), |         # "Dankamania - Easy - 15 Points": GrinchLocationData("Dankamania", ["Supadow Minigames", "Dankamania"], 1503, [GrinchRamData()]), | ||||||
|         # "Dankamania - Hard - 15 Points": GrinchLocationData("Dankamania Supadow", "Supadow Minigames", 1504, [GrinchRamData()]), |         # "Dankamania - Hard - 15 Points": GrinchLocationData("Dankamania", ["Supadow Minigames", "Dankamania"], 1504, [GrinchRamData()]), | ||||||
|         # "Dankamania - Real Tough - 15 Points": GrinchLocationData("Dankamania Supadow", "Supadow Minigames", 1505, [GrinchRamData()]), |         # "Dankamania - Real Tough - 15 Points": GrinchLocationData("Dankamania", ["Supadow Minigames", "Dankamania"], 1505, [GrinchRamData()]), | ||||||
|         # "The Copter Race Contest - Easy": GrinchLocationData("The Copter Race Contest Supadow", "Supadow Minigames", 1506, [GrinchRamData()]), |         # "The Copter Race Contest - Easy": GrinchLocationData("The Copter Race Contest", ["Supadow Minigames", "The Copter Race Contest"], 1506, [GrinchRamData()]), | ||||||
|         # "The Copter Race Contest - Hard": GrinchLocationData("The Copter Race Contest Supadow", "Supadow Minigames", 1507, [GrinchRamData()]), |         # "The Copter Race Contest - Hard": GrinchLocationData("The Copter Race Contest", ["Supadow Minigames", "The Copter Race Contest"], 1507, [GrinchRamData()]), | ||||||
|         # "The Copter Race Contest - Real Tough": GrinchLocationData("The Copter Race Contest Supadow", "Supadow Minigames", 1508, [GrinchRamData()]), |         # "The Copter Race Contest - Real Tough": GrinchLocationData("The Copter Race Contest", ["Supadow Minigames", "The Copter Race Contest"], 1508, [GrinchRamData()]), | ||||||
|         # "Bike Race - 1st Place":  GrinchLocationData("Bike Race", "Supadow Minigames", 1509, [GrinchRamData()]), |         # "Bike Race - 1st Place":  GrinchLocationData("Bike Race", ["Supadow Minigames", "Bike Race"], 1509, [GrinchRamData()]), | ||||||
|         # "Bike Race - Top 2": GrinchLocationData("Bike Race", "Supadow Minigames", 1510, [GrinchRamData()]), |         # "Bike Race - Top 2": GrinchLocationData("Bike Race", ["Supadow Minigames", "Bike Race"], 1510, [GrinchRamData()]), | ||||||
|         # "Bike Race - Top 3": GrinchLocationData("Bike Race", "Supadow Minigames", 1511, [GrinchRamData()]), |         # "Bike Race - Top 3": GrinchLocationData("Bike Race", ["Supadow Minigames", "Bike Race"], 1511, [GrinchRamData()]), | ||||||
| # Sleigh Part Locations | # Sleigh Part Locations | ||||||
|         "Exhaust Pipes in Whoville": GrinchLocationData("Sleigh Room", "Sleigh Ride", 1600, [GrinchRamData(0x0101FB, binary_bit_pos=2)]), |         "WV - Exhaust Pipes": GrinchLocationData("Whoville", ["Sleigh Ride", "Whoville"], 1600, [GrinchRamData(0x0101FB, binary_bit_pos=2)]), | ||||||
|         "Skis in Who Forest": GrinchLocationData("Sleigh Room", "Sleigh Ride", 1601, [GrinchRamData(0x0101FB, binary_bit_pos=3)]), |         "WF - Skis": GrinchLocationData("Who Forest", ["Sleigh Ride", "Who Forest"], 1601, [GrinchRamData(0x0101FB, binary_bit_pos=3)]), | ||||||
|         "Tires in Who Dump": GrinchLocationData("Sleigh Room", "Sleigh Ride", 1602, [GrinchRamData(0x0101FB, binary_bit_pos=4)]), |         "WD - Tires": GrinchLocationData("Who Dump", ["Sleigh Ride", "Who Dump"], 1602, [GrinchRamData(0x0101FB, binary_bit_pos=4)]), | ||||||
|         "Twin-End Tuba in Submarine World": GrinchLocationData("Sleigh Room", "Sleigh Ride", 1603, [GrinchRamData(0x0101FB, binary_bit_pos=6)]), |         "WL - Submarine World - Twin-End Tuba": GrinchLocationData("Submarine World", ["Sleigh Ride", "Who Lake", "South Shore"], 1603, [GrinchRamData(0x0101FB, binary_bit_pos=6)]), | ||||||
|         "GPS in Who Lake": GrinchLocationData("Sleigh Room", "Sleigh Ride", 1604, [GrinchRamData(0x0101FB, binary_bit_pos=5)]), |         "WL - South Shore - GPS": GrinchLocationData("Who Lake", ["Sleigh Ride", "Who Lake", "Submarine World"], 1604, [GrinchRamData(0x0101FB, binary_bit_pos=5)]), | ||||||
|  | # Mount Crumpit Locations | ||||||
|  |         "MC - 1st Crate Squashed": GrinchLocationData("Mount Crumpit", ["Mount Crumpit"], 1700, [GrinchRamData(0x095343, value=1)]), | ||||||
|  |         "MC - 2nd Crate Squashed": GrinchLocationData("Mount Crumpit", ["Mount Crumpit"], 1701, [GrinchRamData(0x095343, value=2)]), | ||||||
|  |         "MC - 3rd Crate Squashed": GrinchLocationData("Mount Crumpit", ["Mount Crumpit"], 1702, [GrinchRamData(0x095343, value=3)]), | ||||||
|  |         "MC - 4th Crate Squashed": GrinchLocationData("Mount Crumpit", ["Mount Crumpit"], 1703, [GrinchRamData(0x095343, value=4)]), | ||||||
|  |         "MC - 5th Crate Squashed": GrinchLocationData("Mount Crumpit", ["Mount Crumpit"], 1704, [GrinchRamData(0x095343, value=5)]), | ||||||
| } | } | ||||||
|  |  | ||||||
| def grinch_locations_to_id() -> dict[str,int]: | def grinch_locations_to_id() -> dict[str,int]: | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| from dataclasses import dataclass | from dataclasses import dataclass | ||||||
|  |  | ||||||
| from Options import FreeText, NumericOption, Toggle, DefaultOnToggle, Choice, TextChoice, Range, NamedRange, OptionList, \ | from Options import FreeText, NumericOption, Toggle, DefaultOnToggle, Choice, TextChoice, Range, NamedRange, OptionList, \ | ||||||
|     PerGameCommonOptions |     PerGameCommonOptions, OptionSet | ||||||
|  |  | ||||||
| class StartingArea(Choice): | class StartingArea(Choice): | ||||||
|     """ |     """ | ||||||
| @@ -21,7 +21,7 @@ class ProgressiveVacuum(Toggle):#DefaultOnToggle | |||||||
|  |  | ||||||
|     Enabled: Whoville > Who Forest > Who Dump > Who Lake |     Enabled: Whoville > Who Forest > Who Dump > Who Lake | ||||||
|     """ |     """ | ||||||
|     display_name = "Progressive Vacuum Access" |     display_name = "Progressive Vacuum Tubes" | ||||||
|  |  | ||||||
| class Missionsanity(Choice): | class Missionsanity(Choice): | ||||||
|     """ |     """ | ||||||
| @@ -39,52 +39,70 @@ class Missionsanity(Choice): | |||||||
|     option_both = 3 |     option_both = 3 | ||||||
|     default = 1 |     default = 1 | ||||||
|  |  | ||||||
| class AnnoyingLocations(DefaultOnToggle): | class ExcludeEnvironments(OptionSet): | ||||||
|     """Makes certain long, annoying, and tedious checks to be excluded [NOT IMPLEMENTED]""" |     """ | ||||||
|     display_name = "Annoying Locations" |     Allows entire environments to be an excluded location to ensure you are not logically required to enter the environment along | ||||||
|  |     with any and all checks that are in that environment too. | ||||||
|  |  | ||||||
|  |     WARNING: Excluding too many environments may cause generation to fail. | ||||||
|  |     [NOT IMPLEMENTED] | ||||||
|  |  | ||||||
|  |     Valid keys: "Whoville", "Who Forest", "Who Dump", "Who Lake", "Post Office", "Clock Tower", "City Hall", | ||||||
|  |                   "Ski Resort", "Civic Center", "Minefield", "Power Plant", "Generator Building", "Scout's Hut", | ||||||
|  |                   "North Shore", "Mayor's Villa", "Sleigh Ride" | ||||||
|  |     """ | ||||||
|  |     display_name = "Exclude Environments" | ||||||
|  |     valid_keys = {"Whoville", "Who Forest", "Who Dump", "Who Lake", "Post Office", "Clock Tower", "City Hall", | ||||||
|  |                   "Ski Resort", "Civic Center", "Minefield", "Power Plant", "Generator Building", "Scout's Hut", | ||||||
|  |                   "North Shore", "Mayor's Villa", "Sleigh Ride"} | ||||||
|  |  | ||||||
| class ProgressiveGadget(Toggle):#DefaultOnToggle | class ProgressiveGadget(Toggle):#DefaultOnToggle | ||||||
|     """ |     """ | ||||||
|     Determines whether you get access to a gadget as individual blueprint count [NOT IMPLEMENTED] |     Determines whether you get access to a gadget as individual blueprint count. [NOT IMPLEMENTED] | ||||||
|     """ |     """ | ||||||
|     display_name = "Progressive Gadgets" |     display_name = "Progressive Gadgets" | ||||||
|  |  | ||||||
| class Supadow(Toggle): | class Supadow(Toggle): | ||||||
|     """Enables completing minigames through the Supadows in Mount Crumpit as checks. (9 locations) [NOT IMPLEMENTED]""" |     """Enables completing minigames through the Supadows in Mount Crumpit as checks. NOT IMPLEMENTED]""" | ||||||
|     display_name = "Supadow Minigame Locations" |     display_name = "Supadow Minigames" | ||||||
|  |  | ||||||
|  | class Gifts(Range): | ||||||
|  |     """ | ||||||
|  |     Considers how many gifts must be squashed per check. | ||||||
|  |     Enabling this will also enable squashing all gifts in a region mission along side this. [NOT IMPLEMENTED] | ||||||
|  |     """ | ||||||
|  |     display_name = "Gifts Squashed per Check" | ||||||
|  |     range_start = 0 | ||||||
|  |     range_end = 300 | ||||||
|  |     default = 0 | ||||||
|  |  | ||||||
| class Gifts(Toggle): | class Moverando(Toggle): | ||||||
|     """Missions that require you to squash every present in a level. (4 locations) [NOT IMPLEMENTED]""" |     """Randomizes Grinch's moveset along with randomizing max into the pool. [NOT IMPLEMENTED] | ||||||
|     display_name = "Gift Collection Locations" |     """ | ||||||
|  |     display_name = "Moves Randomized" | ||||||
|  |  | ||||||
| class Movesanity(Toggle): |  | ||||||
|     """Randomizes Grinch's moveset along with randomizing max into the pool. [NOT IMPLEMENTED]""" |  | ||||||
|     display_name = "Movesanity" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class UnlimitedEggs(Toggle): | class UnlimitedEggs(Toggle): | ||||||
|     """Determine whether or not you run out of rotten eggs when you utilize your gadgets.""" |     """Determine whether or not you run out of rotten eggs when you utilize your gadgets.""" | ||||||
|     display_name = "Unlimited Rotten Eggs" |     display_name = "Unlimited Rotten Eggs" | ||||||
|  |  | ||||||
| class RingLinkOption(Toggle): | class RingLinkOption(Toggle): | ||||||
|     """Whenever this is toggled, your ammo is linked with other ringlink-compatible games that also have this enabled. [NOT IMPLEMENTED]""" |     """Whenever this is toggled, your ammo is linked with other ringlink-compatible games that also have this enabled.""" | ||||||
|  |     display_name = "Ring Link" | ||||||
|  |  | ||||||
| class TrapLinkOption(Toggle): | class TrapLinkOption(Toggle): | ||||||
|     """If a trap is sent from Grinch, traps that are compatible with other games are triggered aswell. [NOT IMPLEMENTED]""" |     """If a trap is sent from Grinch, traps that are compatible with other games are triggered as well. [NOT IMPLEMENTED]""" | ||||||
|  |     display_name = "Trap Link" | ||||||
|  |  | ||||||
| @dataclass | @dataclass | ||||||
| class GrinchOptions(PerGameCommonOptions):#DeathLinkMixin | class GrinchOptions(PerGameCommonOptions):#DeathLinkMixin | ||||||
|     starting_area: StartingArea |     starting_area: StartingArea | ||||||
|     progressive_vacuum: ProgressiveVacuum |     progressive_vacuum: ProgressiveVacuum | ||||||
|     missionsanity: Missionsanity |     missionsanity: Missionsanity | ||||||
|     annoying_locations: AnnoyingLocations |     exclude_environments: ExcludeEnvironments | ||||||
|     minigamesanity: Supadow |  | ||||||
|     progressive_gadget: ProgressiveGadget |     progressive_gadget: ProgressiveGadget | ||||||
|     supadow_minigames: Supadow |     supadow_minigames: Supadow | ||||||
|     giftsanity: Gifts |     giftsanity: Gifts | ||||||
|     movesanity: Movesanity |     move_rando: Moverando | ||||||
|     unlimited_eggs: UnlimitedEggs |     unlimited_eggs: UnlimitedEggs | ||||||
|     ring_link: RingLinkOption |     ring_link: RingLinkOption | ||||||
|     trap_link: TrapLinkOption |     trap_link: TrapLinkOption | ||||||
|   | |||||||
| @@ -19,7 +19,7 @@ mainareas_list = [ | |||||||
| subareas_list = [ | subareas_list = [ | ||||||
|     "Post Office", |     "Post Office", | ||||||
|     "City Hall", |     "City Hall", | ||||||
|     "Countdown to X-Mas Tower", |     "Clock Tower", | ||||||
|     "Ski Resort", |     "Ski Resort", | ||||||
|     "Civic Center", |     "Civic Center", | ||||||
|     "Minefield", |     "Minefield", | ||||||
| @@ -83,12 +83,12 @@ def connect_regions(world: "GrinchWorld"): | |||||||
|     grinchconnect(world, "Mount Crumpit", "Who Dump") |     grinchconnect(world, "Mount Crumpit", "Who Dump") | ||||||
|     grinchconnect(world, "Mount Crumpit", "Who Lake") |     grinchconnect(world, "Mount Crumpit", "Who Lake") | ||||||
|     grinchconnect(world, "Mount Crumpit", "Sleigh Room") |     grinchconnect(world, "Mount Crumpit", "Sleigh Room") | ||||||
|     grinchconnect(world, "Mount Crumpit", "Spin N' Win Supadow") |     grinchconnect(world, "Mount Crumpit", "Spin N' Win") | ||||||
|     grinchconnect(world, "Mount Crumpit", "Dankamania Supadow") |     grinchconnect(world, "Mount Crumpit", "Dankamania") | ||||||
|     grinchconnect(world, "Mount Crumpit", "The Copter Race Contest Supadow") |     grinchconnect(world, "Mount Crumpit", "The Copter Race Contest") | ||||||
|     grinchconnect(world, "Whoville", "Post Office") |     grinchconnect(world, "Whoville", "Post Office") | ||||||
|     grinchconnect(world, "Whoville", "City Hall") |     grinchconnect(world, "Whoville", "City Hall") | ||||||
|     grinchconnect(world, "Whoville", "Countdown to X-Mas Clock Tower") |     grinchconnect(world, "Whoville", "Clock Tower") | ||||||
|     grinchconnect(world, "Who Forest", "Ski Resort") |     grinchconnect(world, "Who Forest", "Ski Resort") | ||||||
|     grinchconnect(world, "Who Forest", "Civic Center") |     grinchconnect(world, "Who Forest", "Civic Center") | ||||||
|     grinchconnect(world, "Who Dump", "Minefield") |     grinchconnect(world, "Who Dump", "Minefield") | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ import Utils | |||||||
| from BaseClasses import CollectionState | from BaseClasses import CollectionState | ||||||
| from worlds.AutoWorld import World | from worlds.AutoWorld import World | ||||||
| from worlds.generic.Rules import add_rule | from worlds.generic.Rules import add_rule | ||||||
|  | from .Items import * | ||||||
|  |  | ||||||
| #Adds all rules from access_rules_dict to locations | #Adds all rules from access_rules_dict to locations | ||||||
| def set_location_rules(world: World): | def set_location_rules(world: World): | ||||||
| @@ -26,423 +27,453 @@ def interpret_rule(rule_set: list[list[str]], player: int): | |||||||
|  |  | ||||||
|     access_list: list[Callable[[CollectionState], bool]] = [] |     access_list: list[Callable[[CollectionState], bool]] = [] | ||||||
|     for item_set in rule_set: |     for item_set in rule_set: | ||||||
|         access_list.append(lambda state: state.has_all(item_set, player)) |         access_list.append(lambda state, items=tuple(item_set): state.has_all(items, player)) | ||||||
|     return access_list |     return access_list | ||||||
|  |  | ||||||
|     #Each item in the list is a separate list of rules. Each separate list is just an "OR" condition. |     #Each item in the list is a separate list of rules. Each separate list is just an "OR" condition. | ||||||
| rules_dict: dict[str,list[list[str]]] = { | rules_dict: dict[str,list[list[str]]] = { | ||||||
|     "Enter Whoville": [ |     "WV - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter the Post Office": [ |     "WV - Post Office - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter the Town Hall": [ |     "WV - City Hall - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter the Countdown-To-Xmas Clock Tower": [ |     "WV - Clock Tower - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter Who Forest": [ |     "WF - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter the Ski Resort": [ |     "WF - Ski Resort - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter the Civic Center": [ |     "WF - Civic Center - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter Who Dump": [ |     "WD - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter the Minefield": [ |     "WD - Minefield - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter the Power Plant": [ |     "WD - Power Plant - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter the Generator Building": [ |     "WD - Generator Building - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter Who Lake": [ |     "WL - South Shore - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter the Submarine World": [ |     "WL - Submarine World - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter the Scout's Hut": [ |     "WL - Scout's Hut - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter the North Shore": [ |     "WL - North Shore - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Enter the Mayor's Villa": [ |     "WL - Mayor's Villa - First Visit": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Shuffling The Mail": [ |     "WV - Post Office - Shuffling The Mail": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Smashing Snowmen": [ |     "WV - Smashing Snowmen": [ | ||||||
|  |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [PC] | ||||||
|  |     ], | ||||||
|  |     "WV - Painting The Mayor's Posters": [ | ||||||
|  |         [PB] | ||||||
|  |     ], | ||||||
|  |     "WV - Launching Eggs Into Houses": [ | ||||||
|  |         [REL] | ||||||
|  |     ], | ||||||
|  |     "WV - City Hall - Modifying The Mayor's Statue": [ | ||||||
|  |         [ST] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [ST, SN], | ||||||
|  |         # [ST, SS] | ||||||
|  |     ], | ||||||
|  |     "WV - Clock Tower - Advancing The Countdown-To-Xmas Clock": [ | ||||||
|  |         [HMR, RS] | ||||||
|  |     ], | ||||||
|  |     "WV - Squashing All Gifts": [ | ||||||
|  |         [GC, SS, REL, WC, RS] | ||||||
|  |     ], | ||||||
|  |     "WF - Making Xmas Trees Droop": [ | ||||||
|  |         [REL] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [REL, BB] | ||||||
|  |     ], | ||||||
|  |     "WF - Sabotaging Snow Cannon With Glue": [ | ||||||
|  |         [GB, RS], | ||||||
|  |         [GB, GC] | ||||||
|  |     ], | ||||||
|  |     "WF - Putting Beehives In Cabins": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WF - Ski Resort - Sliming The Mayor's Skis": [ | ||||||
|  |         [SS, REL] | ||||||
|  |     ], | ||||||
|  |     "WF - Civic Center - Replacing The Candles On The Cake With Fireworks": [ | ||||||
|  |         [REL, GC], | ||||||
|  |         [REL, OCD, RS] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [REL, GC], | ||||||
|  |         # [REL, OCD, RS, SN], | ||||||
|  |         # [REL, OCD, RS, SS] | ||||||
|  |     ], | ||||||
|  |     "WF - Squashing All Gifts": [ | ||||||
|  |         [GC, CCAC, SS, REL], | ||||||
|  |         [OCD, RS, CCAC, SS, REL] | ||||||
|  |     ], | ||||||
|  |     "WD - Stealing Food From Birds": [ | ||||||
|  |         [RS, REL] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [RS, REL, PC] | ||||||
|  |     ], | ||||||
|  |     "WD - Feeding The Computer With Robot Parts": [ | ||||||
|  |         [RS, REL] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [RS, REL, PC] | ||||||
|  |     ], | ||||||
|  |     "WD - Infesting The Mayor's House With Rats": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [REL, RS, PC], | ||||||
|  |         # [REL, GC, PC] | ||||||
|  |     ], | ||||||
|  |     "WD - Conducting The Stinky Gas To Who-Bris' Shack": [ | ||||||
|  |         [RS, REL] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [RS, REL, PC] | ||||||
|  |     ], | ||||||
|  |     "WD - Minefield - Shaving Who Dump Guardian": [ | ||||||
|  |         [SC, GC], | ||||||
|  |         [SC, SS, RS] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [SC, GC, SN], | ||||||
|  |         # [SC, SS, RS, SN] | ||||||
|  |     ], | ||||||
|  |     "WD - Generator Building - Short-Circuiting Power-Plant": [ | ||||||
|  |         [REL, GC], | ||||||
|  |         [REL, OCD, SS, RS] | ||||||
|  |     ], | ||||||
|  |     "WD - Squashing All Gifts": [ | ||||||
|  |         [GC, RS, SS, REL], | ||||||
|  |         [OCD, RS, SS, REL] | ||||||
|  |     ], | ||||||
|  |     "WL - South Shore - Putting Thistles In Shorts": [ | ||||||
|  |         [REL, OCD], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WL - South Shore - Sabotaging The Tents": [ | ||||||
|  |         [OCD, RS], | ||||||
|  |         [GC] | ||||||
|  |     ], | ||||||
|  |     "WL - North Shore - Drilling Holes In Canoes": [ | ||||||
|  |         [DRL] | ||||||
|  |     ], | ||||||
|  |     "WL - Submarine World - Modifying The Marine Mobile": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Painting The Mayor's Posters": [ |     "WL - Mayor's Villa - Hooking The Mayor's Bed To The Motorboat": [ | ||||||
|         ["Painting Bucket"] |         [RP, HK, REL, SCL] | ||||||
|     ], |     ], | ||||||
|     "Launching Eggs Into Houses": [ |     "WL - Squashing All Gifts": [ | ||||||
|         ["Rotten Egg Launcher"] |         [GC, MM, SCL, REL, HK, RP], | ||||||
|  |         [OCD, RS, MM, SCL, REL, HK, RP] | ||||||
|     ], |     ], | ||||||
|     "Modifying The Mayor's Statue": [ |     "WV - Binoculars BP on Post Office Roof": [ | ||||||
|         ["Sculpting Tools"] |  | ||||||
|     ], |  | ||||||
|     "Advancing The Countdown-To-Xmas Clock": [ |  | ||||||
|         ["Hammer", "Rocket Spring"] |  | ||||||
|     ], |  | ||||||
|     "Squashing All Gifts in Whoville": [ |  | ||||||
|         ["Grinch Copter", "Slime Shooter", "Rotten Egg Launcher", "Who Cloak", "Rocket Spring"] |  | ||||||
|     ], |  | ||||||
|     "Making Xmas Trees Droop": [ |  | ||||||
|         ["Rotten Egg Launcher"] |  | ||||||
|     ], |  | ||||||
|     "Sabotaging Snow Cannon With Glue": [ |  | ||||||
|         ["Glue Bucket", "Rocket Spring"], |  | ||||||
|         ["Glue Bucket", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "Putting Beehives In Cabins": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "Sliming The Mayor's Skis": [ |  | ||||||
|         ["Slime Shooter", "Rotten Egg Launcher"] |  | ||||||
|     ], |  | ||||||
|     "Replacing The Candles On The Cake With Fireworks": [ |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"], |  | ||||||
|         ["Rotten Egg Launcher", "Octopus Climbing Device", "Rocket Spring"] |  | ||||||
|     ], |  | ||||||
|     "Squashing All Gifts in Who Forest": [ |  | ||||||
|         ["Grinch Copter", "Cable Car Access Card", "Slime Shooter", "Rotten Egg Launcher"], |  | ||||||
|         ["Octopus Climbing Device", "Rocket Spring", "Cable Car Access Card", "Slime Shooter", "Rotten Egg Launcher"] |  | ||||||
|     ], |  | ||||||
|     "Stealing Food From Birds": [ |  | ||||||
|         ["Rocket Spring", "Rotten Egg Launcher"] |  | ||||||
|     ], |  | ||||||
|     "Feeding The Computer With Robot Parts": [ |  | ||||||
|         ["Rocket Spring", "Rotten Egg Launcher"] |  | ||||||
|     ], |  | ||||||
|     "Infesting The Mayor's House With Rats": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "Conducting The Stinky Gas To Who-Bris' Shack": [ |  | ||||||
|         ["Rocket Spring", "Rotten Egg Launcher"] |  | ||||||
|     ], |  | ||||||
|     "Shaving Who Dump Guardian": [ |  | ||||||
|         ["Scissors", "Grinch Copter"], |  | ||||||
|         ["Scissors", "Slime Shooter", "Rocket Spring"] |  | ||||||
|     ], |  | ||||||
|     "Short-Circuiting Power-Plant": [ |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"], |  | ||||||
|         ["Rotten Egg Launcher", "Octopus Climbing Device", "Slime Shooter", "Rocket Spring"] |  | ||||||
|     ], |  | ||||||
|     "Squashing All Gifts in Who Dump": [ |  | ||||||
|         ["Grinch Copter", "Rocket Spring", "Slime Shooter", "Rotten Egg Launcher"], |  | ||||||
|         ["Octopus Climbing Device", "Rocket Spring", "Slime Shooter", "Rotten Egg Launcher"] |  | ||||||
|     ], |  | ||||||
|     "Putting Thistles In Shorts": [ |  | ||||||
|         ["Rotten Egg Launcher", "Octopus Climbing Device"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "Sabotaging The Tents": [ |  | ||||||
|         ["Octopus Climbing Device", "Rocket Spring"], |  | ||||||
|         ["Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "Drilling Holes In Canoes": [ |  | ||||||
|         ["Drill"] |  | ||||||
|         # ["Drill", "Max"] |  | ||||||
|     ], |  | ||||||
|     "Modifying The Marine Mobile": [ |  | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Hooking The Mayor's Bed To The Motorboat": [ |     "WV - City Hall - Binoculars BP left side of Library": [ | ||||||
|         ["Rope", "Hook", "Rotten Egg Launcher", "Scout Clothes"] |  | ||||||
|     ], |  | ||||||
|     "Squashing All Gifts in Who Lake": [ |  | ||||||
|         ["Grinch Copter", "Marine Mobile", "Scout Clothes", "Rotten Egg Launcher", "Hook", "Rope"], |  | ||||||
|         ["Octopus Climbing Device", "Rocket Spring", "Marine Mobile", "Scout Clothes", "Rotten Egg Launcher", "Hook", "Rope"] |  | ||||||
|     ], |  | ||||||
|     "Binoculars Blueprint - Post Office Roof": [ |  | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Binoculars Blueprint - City Hall Library - Left Side": [ |     "WV - City Hall - Binoculars BP front side of Library": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Binoculars Blueprint - City Hall Library - Front Side": [ |     "WV - City Hall - Binoculars BP right side of Library": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Binoculars Blueprint - City Hall Library - Right Side": [ |     "WV - REL BP left of City Hall": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "REL Blueprint - Outside City Hall": [ |     "WV - REL BP left of Clock Tower": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "REL Blueprint - Outside Clock Tower": [ |     "WV - Post Office - REL BP inside Silver Room": [ | ||||||
|  |         [WC] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [WC, MX] | ||||||
|  |     ], | ||||||
|  |     "WV - Post Office - REL BP at Entrance Door after Mission Completion": [ | ||||||
|  |         [WC] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [WC, MX] | ||||||
|  |     ], | ||||||
|  |     "WF - RS BP behind Vacuum Tube": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "REL Blueprint - Post Office - Inside Silver Room": [ |     "WF - RS BP in front of 2nd House near Vacuum Tube": [ | ||||||
|         ["Who Cloak"] |  | ||||||
|         # ["Who Cloak", "Max"] |  | ||||||
|     ], |  | ||||||
|     "REL Blueprint - Post Office - After Mission Completion": [ |  | ||||||
|         ["Who Cloak"] |  | ||||||
|         # ["Who Cloak", "Max"] |  | ||||||
|     ], |  | ||||||
|     "RS Blueprint - Behind Vacuum": [ |  | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "RS Blueprint - Front of 2nd House near entrance": [ |     "WF - RS BP near Tree House on Ground": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "RS Blueprint - Near Tree House on Ground": [ |     "WF - RS BP behind Cable Car House": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "RS Blueprint - Near Cable Car House": [ |     "WF - RS BP near Who Snowball in Cave": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "RS Blueprint - Near Who Snowball in Cave": [ |     "WF - RS BP on Branch Platform closest to Glue Cannon": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "RS Blueprint - Branch Platform Closest to Glue Cannon": [ |     "WF - RS BP on Branch Platform Near Beast": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "RS Blueprint - Branch Platform Near Beast": [ |     "WF - RS BP on Branch Platform Elevated next to House": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "RS Blueprint - Branch Platform Ledge Grab House": [ |     "WF - RS BP on Tree House": [ | ||||||
|  |         [REL], | ||||||
|  |         [GC] | ||||||
|  |     ], | ||||||
|  |     "WF - SS BP in Branch Platform Elevated House": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WF - SS BP in Branch Platform House next to Beast": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WF - SS BP in House in front of Civic Center Cave": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WF - SS BP in House next to Tree House": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WF - SS BP in House across from Tree House": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WF - SS BP in 2nd House near Vacuum Tube Right Side": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WF - SS BP in 2nd House near Vacuum Tube Left Side": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WF - SS BP in 2nd House near Vacuum Tube inbetween Blueprints": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WF - SS BP in House near Vacuum Tube": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WD - OCD BP inside Middle Pipe": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC], | ||||||
|  |         [SS, RS], | ||||||
|  |         [SS, GC] | ||||||
|  |     ], | ||||||
|  |     "WD - OCD BP inside Right Pipe": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WD - OCD BP in Vent to Mayor's House": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC] | ||||||
|  |     ], | ||||||
|  |     "WD - OCD BP inside Left Pipe": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC], | ||||||
|  |         [SS, RS], | ||||||
|  |         [SS, GC] | ||||||
|  |     ], | ||||||
|  |     "WD - OCD BP near Right Side of Power Plant Wall": [ | ||||||
|  |         [REL, RS], | ||||||
|  |         [REL, GC], | ||||||
|  |         [SS, RS], | ||||||
|  |         [SS, GC] | ||||||
|  |     ], | ||||||
|  |     "WD - OCD BP near Who-Bris' Shack": [ | ||||||
|  |         [REL, RS] | ||||||
|  |     ], | ||||||
|  |     "WD - Minefield - OCD BP on Left Side of House": [ | ||||||
|  |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [REL, GC], | ||||||
|  |         # [REL, SS, RS] | ||||||
|  |         # [MX] | ||||||
|  |     ], | ||||||
|  |     "WD - Minefield - OCD BP on Right Side of Shack": [ | ||||||
|  |         [GC], | ||||||
|  |         [SS, RS] | ||||||
|  |     ], | ||||||
|  |     "WD - Minefield - OCD BP inside Guardian's House": [ | ||||||
|  |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [REL, GC], | ||||||
|  |         # [REL, SS, RS] | ||||||
|  |         # [MX] | ||||||
|  |     ], | ||||||
|  |     "WL - South Shore - MM BP on Bridge to Scout's Hut": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "RS Blueprint - On Tree House": [ |     "WL - South Shore - MM BP across from Tent near Porcupine": [ | ||||||
|         ["Rotten Egg Launcher"], |  | ||||||
|         ["Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "SS Blueprint - Branch Platform Elevated House": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "SS Blueprint - Branch Platform House next to Beast": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "SS Blueprint - House near Civic Center Cave": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "SS Blueprint - House next to Tree House": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "SS Blueprint - House across from Tree House": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "SS Blueprint - 2nd House near entrance right side": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "SS Blueprint - 2nd House near entrance left side": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "SS Blueprint - 2nd House near entrance inbetween blueprints": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "SS Blueprint - House near entrance": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "OCD Blueprint - Middle Pipe": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"], |  | ||||||
|         ["Slime Shooter", "Rocket Spring"], |  | ||||||
|         ["Slime Shooter", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "OCD Blueprint - Right Pipe": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "OCD Blueprint - Mayor's House Rat Vent": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "OCD Blueprint - Left Pipe": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"], |  | ||||||
|         ["Slime Shooter", "Rocket Spring"], |  | ||||||
|         ["Slime Shooter", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "OCD Blueprint - Near Power Plant Wall on right side": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"], |  | ||||||
|         ["Slime Shooter", "Rocket Spring"], |  | ||||||
|         ["Slime Shooter", "Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "OCD Blueprint - Near Who-Bris' Shack": [ |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"] |  | ||||||
|     ], |  | ||||||
|     "OCD Blueprint - Guardian's House - Left Side": [ |  | ||||||
|         [] |  | ||||||
|         # ["Rotten Egg Launcher", "Grinch Copter"], |  | ||||||
|         # ["Rotten Egg Launcher", "Slime Shooter", "Rocket Spring"] |  | ||||||
|         # ["Max"] |  | ||||||
|     ], |  | ||||||
|     "OCD Blueprint - Guardian's House - Right Side": [ |  | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"], |  | ||||||
|         ["Slime Shooter", "Rocket Spring"] |  | ||||||
|     ], |  | ||||||
|     "OCD Blueprint - Inside Guardian's House": [ |  | ||||||
|         [] |  | ||||||
|         # ["Rotten Egg Launcher", "Grinch Copter"], |  | ||||||
|         # ["Rotten Egg Launcher", "Slime Shooter", "Rocket Spring"] |  | ||||||
|         # ["Max"] |  | ||||||
|     ], |  | ||||||
|     "MM Blueprint - South Shore - Bridge to Scout's Hut": [ |  | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "MM Blueprint - South Shore - Tent near Porcupine": [ |     "WL - South Shore - MM BP near Outhouse": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "MM Blueprint - South Shore - Near Outhouse": [ |     "WL - South Shore - MM BP near Hill Bridge": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "MM Blueprint - South Shore - Near Hill Bridge": [ |     "WL - South Shore - MM BP on Scout's Hut Roof": [ | ||||||
|  |         [RS], | ||||||
|  |         [GC] | ||||||
|  |     ], | ||||||
|  |     "WL - South Shore - MM BP on Grass Platform": [ | ||||||
|  |         [RS], | ||||||
|  |         [GC] | ||||||
|  |     ], | ||||||
|  |     "WL - South Shore - MM BP across Zipline Platform": [ | ||||||
|  |         [RS, OCD], | ||||||
|  |         [GC] | ||||||
|  |     ], | ||||||
|  |     "WL - South Shore - MM BP behind Summer Beast": [ | ||||||
|  |         [REL, OCD], | ||||||
|  |         [GC] | ||||||
|  |     ], | ||||||
|  |     "WL - North Shore - MM BP below Bridge": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "MM Blueprint - South Shore - Scout's Hut Roof": [ |     "WL - North Shore - MM BP behind Skunk Hut": [ | ||||||
|         ["Rocket Spring"], |  | ||||||
|         ["Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "MM Blueprint - South Shore - Grass Platform": [ |  | ||||||
|         ["Rocket Spring"], |  | ||||||
|         ["Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "MM Blueprint - South Shore - Zipline by Beast": [ |  | ||||||
|         ["Rocket Spring", "Octopus Climbing Device"], |  | ||||||
|         ["Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "MM Blueprint - South Shore - Behind Summer Beast": [ |  | ||||||
|         ["Rotten Egg Launcher", "Octopus Climbing Device"], |  | ||||||
|         ["Grinch Copter"] |  | ||||||
|     ], |  | ||||||
|     "MM Blueprint - South Shore - Below Bridge": [ |  | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "MM Blueprint - North Shore - Below Bridge": [ |     "WL - North Shore - MM BP inside Skunk Hut": [ | ||||||
|  |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [MX] | ||||||
|  |     ], | ||||||
|  |     "WL - North Shore - MM BP inside House's Fence": [ | ||||||
|  |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [MX] | ||||||
|  |     ], | ||||||
|  |     "WL - North Shore - MM BP inside Boulder Box near Bridge": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "MM Blueprint - North Shore - Behind Skunk Hut": [ |     "WL - North Shore - MM BP inside Boulder Box behind Skunk Hut": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "MM Blueprint - North Shore - Inside Skunk Hut": [ |     "WL - North Shore - MM BP inside Drill House": [ | ||||||
|         [] |  | ||||||
|         # ["Max"] |  | ||||||
|     ], |  | ||||||
|     "MM Blueprint - North Shore - Fenced in Area": [ |  | ||||||
|         [] |  | ||||||
|         # ["Max"] |  | ||||||
|     ], |  | ||||||
|     "MM Blueprint - North Shore - Boulder Box near Bridge": [ |  | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "MM Blueprint - North Shore - Boulder Box behind Skunk Hut": [ |     "WL - North Shore - MM BP on Crow Platform near Drill House": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "MM Blueprint - North Shore - Inside Drill House": [ |     "WV - City Hall - GC BP in Safe Room": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "MM Blueprint - North Shore - Crow Platform near Drill House": [ |     "WV - City Hall - GC BP in Statue Room": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Whoville City Hall - Safe Room": [ |     "WV - Clock Tower - GC BP in Bedroom": [ | ||||||
|  |         [RS] | ||||||
|  |         # "move_rando" | ||||||
|  |     #   [MX, RS] | ||||||
|  |     ], | ||||||
|  |     "WV - Clock Tower - GC BP in Bell Room": [ | ||||||
|  |         [RS] | ||||||
|  |     ], | ||||||
|  |     "WF - Ski Resort - GC BP inside Dog's Fence": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Whoville City Hall - Statue Room": [ |     "WF - Ski Resort - GC BP in Max Cave": [ | ||||||
|         [] |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [MX] | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Whoville Clock Tower - Before Bells": [ |     "WF - Civic Center - GC BP on Left Side in Bat Cave Wall": [ | ||||||
|         ["Rocket Spring"] |         [GC], | ||||||
|     #   ["Max", "Rocket Spring"] |         [OCD, RS] | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Whoville Clock Tower - After Bells": [ |     "WF - Civic Center - GC BP in Frozen Ice": [ | ||||||
|         ["Rocket Spring"] |         [REL, GC], | ||||||
|  |         [REL, OCD, RS], | ||||||
|  |         [SS, GC], | ||||||
|  |         [SS, OCD, RS] | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Who Forest Ski Resort - Inside Dog's Fence": [ |     "WD - Power Plant - GC BP in Max Cave": [ | ||||||
|         [] |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [MX] | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Who Forest Ski Resort - Max Cave": [ |     "WD - Power Plant - GC BP After First Gate": [ | ||||||
|         [] |         [REL, RS], | ||||||
|         # ["Max"] |         [GC] | ||||||
|  |         # "move_rando" | ||||||
|  |     #   [MX, REL, RS] | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Who Forest Civic Center - Climb across Bat Cave wall": [ |     "WD - Generator Building - GC BP on the Highest Platform": [ | ||||||
|         ["Grinch Copter"], |         [REL, GC], | ||||||
|         ["Octopus Climbing Device", "Rocket Spring"] |         [REL, OCD, SS, RS] | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Who Forest Civic Center - Shoot Icicle in Bat Entrance": [ |     "WD - Generator Building - GC BP at the Entrance after Mission Completion": [ | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"], |         [REL, GC], | ||||||
|         ["Rotten Egg Launcher", "Octopus Climbing Device", "Rocket Spring"], |         [REL, OCD, SS, RS] | ||||||
|         ["Slime Shooter", "Grinch Copter"], |  | ||||||
|         ["Slime Shooter", "Octopus Climbing Device", "Rocket Spring"] |  | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Who Dump Power Plant - Max Cave": [ |     "WL - Submarine World - GC BP Just Below Water Surface": [ | ||||||
|         [] |         [MM] | ||||||
|         # ["Max"] |  | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Who Dump Power Plant - After First Gate": [ |     "WL - Submarine World - GC BP Underwater": [ | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"], |         [MM] | ||||||
|         ["Grinch Copter"] |  | ||||||
|     #   ["Max", "Rotten Egg Launcher", "Rocket Spring"] |  | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Who Dump Generator Building - Before Mission": [ |     "WL - Mayor's Villa - GC BP on Tree Branch": [ | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"], |         [GC], | ||||||
|         ["Rotten Egg Launcher", "Octopus Climbing Device", "Slime Shooter", "Rocket Spring"] |         [REL, RS] | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Who Dump Generator Building - After Mission": [ |     "WL - Mayor's Villa - GC BP in Pirate's Cave": [ | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"], |         [GC], | ||||||
|         ["Rotten Egg Launcher", "Octopus Climbing Device", "Slime Shooter", "Rocket Spring"] |         [REL, RS] | ||||||
|     ], |     ], | ||||||
|     "GC Blueprint - Who Lake South Shore - Submarine World - Above Surface": [ |     "MC - Sleigh Ride - Stealing All Gifts": [ | ||||||
|         ["Marine Mobile"] |  | ||||||
|     ], |  | ||||||
|     "GC Blueprint - Who Lake South Shore - Submarine World - Underwater": [ |  | ||||||
|         ["Marine Mobile"] |  | ||||||
|     ], |  | ||||||
|     "GC Blueprint - Who Lake North Shore - Mayor's Villa - Tree Branch": [ |  | ||||||
|         ["Grinch Copter"], |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"] |  | ||||||
|     ], |  | ||||||
|     "GC Blueprint - Who Lake North Shore - Mayor's Villa - Cave": [ |  | ||||||
|         ["Grinch Copter"], |  | ||||||
|         ["Rotten Egg Launcher", "Rocket Spring"] |  | ||||||
|     ], |  | ||||||
|     "Stealing All Gifts": [ |  | ||||||
|         # ["Exhaust Pipes", "Tires", "Skis", "Twin-End Tuba"] |         # ["Exhaust Pipes", "Tires", "Skis", "Twin-End Tuba"] | ||||||
|         ["Rotten Egg Launcher", "Who Forest Vacuum Access", "Who Dump Vacuum Access", "Who Lake Vacuum Access", "Rocket Spring", "Marine Mobile"] |         [REL, WV, WF, WD, WL, RS, MM] | ||||||
|     ], |     ], | ||||||
|     "Neutralizing Santa": [ |     "MC - Sleigh Ride - Neutralizing Santa": [ | ||||||
|         # ["Exhaust Pipes", "Tires", "Skis", "Twin-End Tuba"] |         # ["Exhaust Pipes", "Tires", "Skis", "Twin-End Tuba"] | ||||||
|         ["Rotten Egg Launcher", "Who Forest Vacuum Access", "Who Dump Vacuum Access", "Who Lake Vacuum Access", "Rocket Spring", "Marine Mobile"] |         [REL, WV, WF, WD, WL, RS, MM] | ||||||
|     ], |     ], | ||||||
|     "Heart of Stone - Whoville's Post Office": [ |     "WV - Post Office - Heart of Stone": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Heart of Stone - Who Forest's Ski Resort": [ |     "WF - Ski Resort - Heart of Stone": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Heart of Stone - Who Dump's Minefield": [ |     "WD - Minefield - Heart of Stone": [ | ||||||
|         ["Grinch Copter"], |         [GC], | ||||||
|         ["Rotten Egg Launcher", "Slime Shooter", "Rocket Spring"] |         [REL, SS, RS] | ||||||
|     ], |     ], | ||||||
|     "Heart of Stone - Who Lake's North Shore": [ |     "WL - North Shore - Heart of Stone": [ | ||||||
|         [] |         [] | ||||||
|         # ["Max"] |         # "move_rando" | ||||||
|  |         # [MX] | ||||||
|     ], |     ], | ||||||
|     "Spin N' Win - Easy": [ |     "Spin N' Win - Easy": [ | ||||||
|         [] |         [] | ||||||
| @@ -480,99 +511,154 @@ rules_dict: dict[str,list[list[str]]] = { | |||||||
|     "Bike Race - Top 3": [ |     "Bike Race - Top 3": [ | ||||||
|         [] |         [] | ||||||
|     ], |     ], | ||||||
|     "Exhaust Pipes in Whoville": [ |     "WV - Exhaust Pipes": [ | ||||||
|         ["Rotten Egg Launcher"] |         [WV, REL, SR] | ||||||
|     ], |     ], | ||||||
|     "Skis in Who Forest": [ |     "WF - Skis": [ | ||||||
|         ["Who Forest Vacuum Access"] |         [WF, SR] | ||||||
|     ], |     ], | ||||||
|     "Tires in Who Dump": [ |     "WD - Tires": [ | ||||||
|         ["Who Dump Vacuum Access", "Rocket Spring", "Rotten Egg Launcher"] |         [WD, RS, REL, SR] | ||||||
|     ], |     ], | ||||||
|     "Twin-End Tuba in Submarine World": [ |     "WL - Submarine World - Twin-End Tuba": [ | ||||||
|         ["Who Lake Vacuum Access", "Marine Mobile"] |         [WL, MM, SR] | ||||||
|     ], |     ], | ||||||
|     "GPS in Who Lake": [ |     "WL - South Shore - GPS": [ | ||||||
|         ["Who Lake Vacuum Access", "Rotten Egg Launcher"] |         [WL, REL, SR] | ||||||
|     ], |     ], | ||||||
|  |     "MC - 1st Crate Squashed": [ | ||||||
|  |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [PC] | ||||||
|  |     ], | ||||||
|  |     "MC - 2nd Crate Squashed": [ | ||||||
|  |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [PC] | ||||||
|  |     ], | ||||||
|  |     "MC - 3rd Crate Squashed": [ | ||||||
|  |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [PC] | ||||||
|  |     ], | ||||||
|  |     "MC - 4th Crate Squashed": [ | ||||||
|  |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [PC] | ||||||
|  |     ], | ||||||
|  |     "MC - 5th Crate Squashed": [ | ||||||
|  |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [PC] | ||||||
|  |     ] | ||||||
|  |     # "Green Present": [ | ||||||
|  |     #     [] | ||||||
|  |     # ], | ||||||
|  |     # "Red Present": [ | ||||||
|  |     #     [] | ||||||
|  |     # ], | ||||||
|  |     # "Pink Present": [ | ||||||
|  |     #     [REL], | ||||||
|  |     #     [move_rando] | ||||||
|  |     #     [PC] | ||||||
|  |     # ], | ||||||
|  |     # "Yellow Present": [ | ||||||
|  |     #     [] | ||||||
|  |     #     "move_rando" | ||||||
|  |     #     [PC] | ||||||
|  |     # ] | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| access_rules_dict: dict[str,list[list[str]]] = { | access_rules_dict: dict[str,list[list[str]]] = { | ||||||
|     "Whoville": [ |     "Whoville": [ | ||||||
|         [] |         [WV] | ||||||
|     ], |     ], | ||||||
|     "Post Office": [ |     "Post Office": [ | ||||||
|         ["Who Cloak"] |         [WC] | ||||||
|     ], |     ], | ||||||
|     "City Hall": [ |     "City Hall": [ | ||||||
|         ["Rotten Egg Launcher"] |         [REL] | ||||||
|     ], |     ], | ||||||
|     "Countdown to X-Mas Clock Tower": [ |     "Clock Tower": [ | ||||||
|         [] |         [] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [SN] | ||||||
|     ], |     ], | ||||||
|     "Who Forest": [ |     "Who Forest": [ | ||||||
|         ["Who Forest Vacuum Access"], |         [WF], | ||||||
|         # ["Progressive Vacuum Access": 1] |         # [VT: 1] | ||||||
|     ], |     ], | ||||||
|     "Ski Resort": [ |     "Ski Resort": [ | ||||||
|         ["Cable Car Access Card"] |         [CCAC] | ||||||
|     ], |     ], | ||||||
|     "Civic Center": [ |     "Civic Center": [ | ||||||
|         ["Grinch Copter"], |         [GC], | ||||||
|         ["Octopus Climbing Device"] |         [OCD] | ||||||
|     ], |     ], | ||||||
|     "Who Dump": [ |     "Who Dump": [ | ||||||
|         ["Who Dump Vacuum Access"], |         [WD], | ||||||
|         # ["Progressive Vacuum Access": 2] |         # [VT: 2] | ||||||
|     ], |     ], | ||||||
|     "Minefield": [ |     "Minefield": [ | ||||||
|         ["Rotten Egg Launcher", "Slime Shooter", "Rocket Spring"], |         [REL, RS], | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"] |         [REL, GC] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [REL, RS, PC], | ||||||
|  |         # [REL, GC, PC] | ||||||
|     ], |     ], | ||||||
|     "Power Plant": [ |     "Power Plant": [ | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"], |         [REL, GC], | ||||||
|         ["Slime Shooter", "Grinch Copter"], |         [SS, GC], | ||||||
|         ["Rotten Egg Launcher", "Octopus Climbing Device", "Slime Shooter", "Rocket Spring"] |         [REL, OCD, SS, RS] | ||||||
|  |         # "move_rando" | ||||||
|  |         # [REL, GC, PC], | ||||||
|  |         # [SS, GC, PC], | ||||||
|  |         # [REL, OCD, SS, RS, PC] | ||||||
|     ], |     ], | ||||||
|     "Generator Building": [ |     "Generator Building": [ | ||||||
|         ["Rotten Egg Launcher", "Grinch Copter"], |         [REL, GC], | ||||||
|         ["Rotten Egg Launcher", "Octopus Climbing Device", "Slime Shooter", "Rocket Spring"] |         [REL, OCD, SS, RS] | ||||||
|     ], |     ], | ||||||
|     "Who Lake": [ |     "Who Lake": [ | ||||||
|         ["Who Lake Vacuum Access"], |         [WL], | ||||||
|         # ["Progressive Vacuum Access": 3] |         # [VT: 3] | ||||||
|     ], |     ], | ||||||
|     "Scout's Hut": [ |     "Scout's Hut": [ | ||||||
|         ["Grinch Copter"], |         [GC], | ||||||
|         ["Rocket Spring"] |         [RS] | ||||||
|     ], |     ], | ||||||
|     "North Shore": [ |     "North Shore": [ | ||||||
|         ["Scout Clothes"] |         [SCL] | ||||||
|     ], |     ], | ||||||
|     "Mayor's Villa": [ |     "Mayor's Villa": [ | ||||||
|         ["Scout Clothes"] |         [SCL] | ||||||
|     ], |     ], | ||||||
|     "Submarine World": [ |     "Submarine World": [ | ||||||
|         ["Marine Mobile"] |         [MM] | ||||||
|     ], |     ], | ||||||
|     "Sleigh Room": [ |     "Sleigh Room": [ | ||||||
|         ["Sleigh Room Key"] |         [SR] | ||||||
|     ], |     ], | ||||||
|     "Spin N' Win Supadow": [ |     "Spin N' Win": [ | ||||||
|         ["Spin N' Win Door Unlock"], |         [] | ||||||
|  |         # ["Spin N' Win Door Unlock"], | ||||||
|         # ["Progressive Supadow Door Unlock"] |         # ["Progressive Supadow Door Unlock"] | ||||||
|     ], |     ], | ||||||
|     "Dankamania Supadow": [ |     "Dankamania": [ | ||||||
|         ["Dankamania Door Unlock"], |         [] | ||||||
|  |         # ["Dankamania Door Unlock"], | ||||||
|         # ["Progressive Supadow Door Unlock: 2"] |         # ["Progressive Supadow Door Unlock: 2"] | ||||||
|     ], |     ], | ||||||
|     "The Copter Race Contest Supadow": [ |     "The Copter Race Contest": [ | ||||||
|         ["The Copter Race Contest Door Unlock"], |         [] | ||||||
|  |         # ["The Copter Race Contest Door Unlock"], | ||||||
|         # ["Progressive Supadow Door Unlock: 3"] |         # ["Progressive Supadow Door Unlock: 3"] | ||||||
|     ], |     ], | ||||||
|     "Bike Race": [ |     "Bike Race": [ | ||||||
|         ["Bike Race Access"], |         [] | ||||||
|  |         # ["Bike Race Access"], | ||||||
|         # ["Progressive Supadow Door Unlock: 4"] |         # ["Progressive Supadow Door Unlock: 4"] | ||||||
|     ] |     ] | ||||||
| } | } | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| from BaseClasses import Region, Item, ItemClassification | from BaseClasses import Region, Item, ItemClassification | ||||||
| from .Locations import grinch_locations_to_id, grinch_locations, GrinchLocation | from .Locations import grinch_locations_to_id, grinch_locations, GrinchLocation, get_location_names_per_category | ||||||
| from .Items import grinch_items_to_id, GrinchItem, ALL_ITEMS_TABLE, MISC_ITEMS_TABLE | from .Items import grinch_items_to_id, GrinchItem, ALL_ITEMS_TABLE, MISC_ITEMS_TABLE, get_item_names_per_category | ||||||
| from .Regions import connect_regions | from .Regions import connect_regions | ||||||
| from .Rules import set_location_rules | from .Rules import set_location_rules | ||||||
|  |  | ||||||
| @@ -8,27 +8,32 @@ from .Client import * | |||||||
| from typing import ClassVar | from typing import ClassVar | ||||||
|  |  | ||||||
| from worlds.AutoWorld import World | from worlds.AutoWorld import World | ||||||
|  | from Options import OptionError | ||||||
|  |  | ||||||
| from . import Options | from .Options import GrinchOptions | ||||||
| from .Rules import access_rules_dict | from .Rules import access_rules_dict | ||||||
|  |  | ||||||
|  |  | ||||||
| class GrinchWorld(World): | class GrinchWorld(World): | ||||||
|     game: ClassVar[str] = "The Grinch" |     game: ClassVar[str] = "The Grinch" | ||||||
|     options_dataclass = Options.GrinchOptions |     options_dataclass = Options.GrinchOptions | ||||||
|     options = Options.GrinchOptions |     options: Options.GrinchOptions | ||||||
|     topology_present = True #not an open world game, very linear |     topology_present = True #not an open world game, very linear | ||||||
|     item_name_to_id: ClassVar[dict[str,int]] = grinch_items_to_id() |     item_name_to_id: ClassVar[dict[str,int]] = grinch_items_to_id() | ||||||
|     location_name_to_id: ClassVar[dict[str,int]] = grinch_locations_to_id() |     location_name_to_id: ClassVar[dict[str,int]] = grinch_locations_to_id() | ||||||
|     required_client_version = (0, 6, 3) |     required_client_version = (0, 6, 3) | ||||||
|  |     item_name_groups = get_item_names_per_category() | ||||||
|  |     location_name_groups = get_location_names_per_category() | ||||||
|  |  | ||||||
|     def __init__(self, *args, **kwargs): #Pulls __init__ function and takes control from there in BaseClasses.py |     def __init__(self, *args, **kwargs): #Pulls __init__ function and takes control from there in BaseClasses.py | ||||||
|         self.origin_region_name: str = "Mount Crumpit" |         self.origin_region_name: str = "Mount Crumpit" | ||||||
|         super(GrinchWorld, self).__init__(*args, **kwargs) |         super(GrinchWorld, self).__init__(*args, **kwargs) | ||||||
|  |  | ||||||
|     def generate_early(self) -> None: #Special conditions changed before generation occurs |     def generate_early(self) -> None: #Special conditions changed before generation occurs | ||||||
|         pass |         if self.options.ring_link == 1 and self.options.unlimited_eggs == 1: | ||||||
|  |             raise OptionError("Cannot enable both unlimited rotten eggs and ring links. You can only enable one of these at a time." + | ||||||
|  |                                       f"The following player's YAML needs to be fixed: {self.player_name}") | ||||||
|  |  | ||||||
|  |  | ||||||
|     def create_regions(self): #Generates all regions for the multiworld |     def create_regions(self): #Generates all regions for the multiworld | ||||||
|         for region_name in access_rules_dict.keys(): |         for region_name in access_rules_dict.keys(): | ||||||
| @@ -37,7 +42,7 @@ class GrinchWorld(World): | |||||||
|         for location, data in grinch_locations.items(): |         for location, data in grinch_locations.items(): | ||||||
|             region = self.get_region(data.region) |             region = self.get_region(data.region) | ||||||
|             entry = GrinchLocation(self.player, location, region, data) |             entry = GrinchLocation(self.player, location, region, data) | ||||||
|             if location == "Neutralizing Santa": |             if location == "MC - Sleigh Ride - Neutralizing Santa": | ||||||
|                 entry.place_locked_item(Item("Goal", ItemClassification.progression, None, self.player)) |                 entry.place_locked_item(Item("Goal", ItemClassification.progression, None, self.player)) | ||||||
|             region.locations.append(entry) |             region.locations.append(entry) | ||||||
|         connect_regions(self) |         connect_regions(self) | ||||||
| @@ -72,7 +77,7 @@ class GrinchWorld(World): | |||||||
|     def fill_slot_data(self): |     def fill_slot_data(self): | ||||||
|         return { |         return { | ||||||
|             "give_unlimited_eggs": self.options.unlimited_eggs.value, |             "give_unlimited_eggs": self.options.unlimited_eggs.value, | ||||||
|  |             "ring_link": self.options.ring_link.value, | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     def generate_output(self, output_directory: str) -> None: |     def generate_output(self, output_directory: str) -> None: | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								worlds/grinch/archipelago.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								worlds/grinch/archipelago.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | { | ||||||
|  |     "minimum_ap_version": "0.6.3",  | ||||||
|  |     "world_version": "1.3.0", | ||||||
|  |     "authors": ["MarioSpore"], | ||||||
|  |     "version": 7, | ||||||
|  |     "compatible_version": 7, | ||||||
|  |     "game": "The Grinch" | ||||||
|  | } | ||||||
| @@ -1,5 +1,7 @@ | |||||||
| - Credit to Raven-187 & Hacc on gamehacking.org for providing the addresses for various cheat codes. Without them, this  | - Credit to Raven-187 & Hacc on gamehacking.org for providing the addresses for various cheat codes. Without them, this  | ||||||
| would of made RAM searching much more tedious. | would of made RAM searching much more tedious. | ||||||
| - Shoutouts to SomeJakeGuy for basically teaching me how to code in general. | - Shoutouts to SomeJakeGuy for basically teaching me how to code in general along with starting decompilation of the game | ||||||
|  | into motion | ||||||
| - Shoutouts to BootsinSoots for helping with the implementation of the logic rules code itself. | - Shoutouts to BootsinSoots for helping with the implementation of the logic rules code itself. | ||||||
| - Thanks to the Grinch PS1 speedrunning discord community server for encouraging the production of this randomizer. | - Thanks to the Grinch PS1 speedrunning discord community server for encouraging the production of this randomizer. | ||||||
|  | - Credit to Artamiss for the vast majority of BZZ decompilation and general backend coding. | ||||||
| @@ -1,31 +1,44 @@ | |||||||
| # The Grinch - Setup Guide | # The Grinch (PS1) - Setup Guide | ||||||
|  |  | ||||||
| ## Required Software | ## Required Software | ||||||
| - [Archipelago](https://github.com/ArchipelagoMW/Archipelago/releases). Please use version 0.6.2 or later for integrated | - [Archipelago](https://github.com/ArchipelagoMW/Archipelago/releases). Please use version 0.6.3 or later for integrated | ||||||
| BizHawk support. | BizHawk support. | ||||||
| - Legally obtained NTSC Bin ROM file, probably named something like `Grinch, The (USA) (En,Fr,Es).bin`.  | - Legally obtained NTSC Bin ROM file, probably named something like `Grinch, The (USA) (En,Fr,Es).bin`.  | ||||||
| - CUE files may work, but I have not tested this. | The game's CUE file should also work aswell along side the BIN file if you have troubles opening the BIN file. | ||||||
| - [BizHawk](https://tasvideos.org/BizHawk/ReleaseHistory) Version 2.9.1 is supported, but I can't promise if any version is stable or not. | - [BizHawk](https://tasvideos.org/BizHawk/ReleaseHistory) Versions 2.9.1 & 2.10 is supported, but I can't promise if any version is stable or not. As of September 2025, the Grinch may not be compatible with 2.11 yet. | ||||||
| - The latest `grinch.apworld` file. You can find this on the [Releases page](https://github.com/MarioSpore/Grinch-AP/releases/latest). Put this in your `Archipelago/custom_worlds` folder. | - The latest `grinch.apworld` file. You can find this on the [Releases page](https://github.com/MarioSpore/Grinch-AP/releases/latest). Put this in your `Archipelago/custom_worlds` folder. | ||||||
|  | - PSX BIOS Firmware bin file, which is required to run the game through Bizhawk. The file you need should be | ||||||
|  | named something like `SCPH-5501.BIN`. | ||||||
|  |  | ||||||
| ## Configuring your Config (.yaml) file | ## Configuring BizHawk | ||||||
|  | Once you have installed BizHawk, open `EmuHawk.exe` and change the following settings: | ||||||
|  |  | ||||||
| ### What is a config file and why do I need one? | - If you're using BizHawk 2.7 or 2.8, go to `Config > Customize`. On the Advanced tab, switch the Lua Core from | ||||||
|  | `NLua+KopiLua` to `Lua+LuaInterface`, then restart EmuHawk. (If you're using BizHawk 2.9, you can skip this step.) | ||||||
|  | - Under `Config > Customize`, check the "Run in background" option to prevent disconnecting from the client while you're | ||||||
|  | tabbed out of EmuHawk. | ||||||
|  | - Under `Config > Preferred Cores > PSX`, select NymaShock. | ||||||
|  | - Open any PlayStation game in EmuHawk and go to `Config > Controllers…` to configure your inputs. If you can't click | ||||||
|  | `Controllers…`, it's because you need to load a game first. | ||||||
|  | You may need to invert Sensitivity for the up/down axis to -100%. | ||||||
|  | This can be found under Analog Controls through `Config > Controllers…`. | ||||||
|  | Depending on your controller, you may also want to tweak the Deadzone. Something like 6% is recommended for a DualShock 4. | ||||||
|  | - Consider clearing keybinds in `Config > Hotkeys…` if you don't intend to use them. Select the keybind and press Esc to | ||||||
|  | clear it. | ||||||
|  | - You are required to legally obtain a PSX Bios BIN firmware file for the game to be opened. To import this, you go to | ||||||
|  | `Config > Firmware... > Tools` and scrolling until you see the PlayStation tab. You might right click on the bios region | ||||||
|  | and click `Set Customization` or `Import` on the top of the window. Then a window should open, telling you what file to  | ||||||
|  | import, which is the BIN file required. The bios should be recognized if Bizhawk displays a checkmark beside it, saying  | ||||||
|  | the bios version you have is stable and should run without issues. If a bios is already been imported and the game | ||||||
|  | runs fine by itself, you may skip this step. | ||||||
|  |  | ||||||
| See the guide on setting up a basic YAML at the Archipelago setup | ## Generating a Game | ||||||
| guide: [Basic Multiworld Setup Guide](/tutorial/Archipelago/setup/en) |  | ||||||
|  |  | ||||||
| ### Where do I get a config file? | 1. Create your options file (YAML). After installing the `grinch.apworld` file, you can generate a template within the Archipelago Launcher by clicking `Generate Template Settings`. | ||||||
|  | 2. Follow the general Archipelago instructions for [generating a game](https://archipelago.gg/tutorial/Archipelago/setup/en#generating-a-game). | ||||||
| The Player options page on the website allows you to configure your personal | 3. Open `ArchipelagoLauncher.exe` | ||||||
| options and export a config file from them: [The Grinch Player Options Page](../player-options) | 4. Select "BizHawk Client" in the right-side column. On your first time opening BizHawk Client, you will also be asked to | ||||||
|  | locate `EmuHawk.exe` in your BizHawk install. | ||||||
| ### Verifying your config file |  | ||||||
|  |  | ||||||
| If you would like to validate your config file to make sure it works, you may do |  | ||||||
| so on the YAML Validator page: [YAML Validation page](/check) |  | ||||||
|  |  | ||||||
| ## Joining a MultiWorld Game |  | ||||||
|  |  | ||||||
| ### Connect to the Multiserver | ### Connect to the Multiserver | ||||||
|  |  | ||||||
| @@ -37,18 +50,3 @@ script. Navigate to your Archipelago install folder and open `data/lua/connector | |||||||
| To connect the client to the multiserver simply put `<address>:<port>` on the text field on top and | To connect the client to the multiserver simply put `<address>:<port>` on the text field on top and | ||||||
| press enter (if the server uses a password, type in the bottom text field | press enter (if the server uses a password, type in the bottom text field | ||||||
| `/connect <address>:<port> [password]`) | `/connect <address>:<port> [password]`) | ||||||
|  |  | ||||||
| ## Hosting a MultiWorld game |  | ||||||
|  |  | ||||||
| The recommended way to host a game is to use our hosting service. The process is relatively simple: |  | ||||||
|  |  | ||||||
| 1. Collect config files from your players. |  | ||||||
| 2. Upload the config files to the Generate page above. |  | ||||||
|     - Generate page: [WebHost Seed Generation Page](/generate) |  | ||||||
| 3. Wait a moment while the seed is generated. |  | ||||||
| 4. When the seed is generated, you will be redirected to a "Seed Info" page. |  | ||||||
| 5. Click "Create New Room". This will take you to the server page. Provide the link to this page to |  | ||||||
|   your players, so they may download their patch files from there. |  | ||||||
| 6. Note that a link to a MultiWorld Tracker is at the top of the room page. The tracker shows the |  | ||||||
|   progress of all players in the game. Any observers may also be given the link to this page. |  | ||||||
| 7. Once all players have joined, you may begin playing. |  | ||||||
		Reference in New Issue
	
	Block a user