mirror of
				https://github.com/MarioSpore/Grinch-AP.git
				synced 2025-10-21 20:21:32 -06:00 
			
		
		
		
	Compare commits
	
		
			90 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 90ba4fbda7 | ||
|   | 8ff2fb91d4 | ||
|   | ee1190cf12 | ||
|   | b7315a9991 | ||
|   | e76dd67ff6 | ||
|   | a16de9da0a | ||
|   | af0527f9a6 | ||
|   | 04bb867805 | ||
|   | 5cfbf84519 | ||
|   | a00cd0212a | ||
|   | 61885767d5 | ||
|   | b0619d5751 | ||
|   | 769ab01d19 | ||
|   | b918d96294 | ||
|   | dc76761fc8 | ||
|   | c48bca965e | ||
|   | 0e294f53ec | ||
|   | cf7f16d36b | ||
|   | 78bc54d227 | ||
|   | 30fd16bdb8 | ||
|   | 95fb26f20c | ||
|   | 8c07ca8c81 | ||
|   | fc31e2f442 | ||
|   | bfca1df83c | ||
|   | 41e55b169f | ||
|   | e8e9c76eda | ||
|   | ff9c7480db | ||
|   | 5444ea7061 | ||
|   | 069355778d | ||
|   | 9a6f6f7a75 | ||
|   | 7105187ad3 | ||
|   | 98b971a659 | ||
|   | e6430b2f86 | ||
|   | 4a6f4fce4f | ||
|   | 837e651d7b | ||
|   | 4d1d728db1 | ||
|   | 3dc4802be7 | ||
|   | 08e9df66de | ||
|   | 14d7bdba15 | ||
|   | 5eaf551584 | ||
|   | e941e8bdbf | ||
|   | 659ae21fa7 | ||
|   | 032dd8712e | ||
|   | 5caacaac87 | ||
|   | 95e80227e1 | ||
|   | dced197dc4 | ||
|   | 3549e55c59 | ||
|   | 24d1b96b9e | ||
|   | 76b4ff2a6e | ||
|   | d9e300e0fd | ||
|   | 98e2486292 | ||
|   | 044fdaa717 | ||
|   | 922232264d | ||
|   | 92dafd0a73 | ||
|   | a010080371 | ||
|   | 849691b009 | ||
|   | 912c4db021 | ||
|   | d91ade58ee | ||
|   | 90d02672b5 | ||
|   | 8ba0bbc73a | ||
|   | 59e4a6c1e3 | ||
|   | a2b1f885a5 | ||
|   | ceec3ed28b | ||
|   | 7ad1211960 | ||
|   | 3b7a6554ac | ||
|   | a49921392b | ||
|   | f71038d17c | ||
|   | ea4f03118b | ||
|   | 54d99f5b54 | ||
|   | f38a5fbadd | ||
|   | 4123961e81 | ||
|   | 30f800f648 | ||
|   | 2c5cb791a6 | ||
|   | 397693c8a8 | ||
|   | 3541e13f21 | ||
|   | 0f2851e1b3 | ||
|   | b0a9831082 | ||
|   | cf921a8f54 | ||
|   | b622953cd0 | ||
|   | 64cca7fff9 | ||
|   | 1762fefba9 | ||
|   | 7e06efb1d0 | ||
|   | c3ddce5b1a | ||
|   | 3c622eefe4 | ||
|   | 17aebc940a | ||
|   | 110da2b524 | ||
|   | 10fa86d52c | ||
|   | bede861e3d | ||
|   | 694ba4c9bb | ||
|   | a7d5d45d14 | 
| @@ -1,6 +1,7 @@ | |||||||
| <component name="ProjectRunConfigurationManager"> | <component name="ProjectRunConfigurationManager"> | ||||||
|   <configuration default="false" name="Archipelago Unittests" type="tests" factoryName="Unittests"> |   <configuration default="false" name="Archipelago Unittests" type="tests" factoryName="Unittests"> | ||||||
|     <module name="Archipelago" /> |     <module name="Grinch-AP" /> | ||||||
|  |     <option name="ENV_FILES" value="" /> | ||||||
|     <option name="INTERPRETER_OPTIONS" value="" /> |     <option name="INTERPRETER_OPTIONS" value="" /> | ||||||
|     <option name="PARENT_ENVS" value="true" /> |     <option name="PARENT_ENVS" value="true" /> | ||||||
|     <option name="SDK_HOME" value="" /> |     <option name="SDK_HOME" value="" /> | ||||||
|   | |||||||
							
								
								
									
										287
									
								
								worlds/grinch/Client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										287
									
								
								worlds/grinch/Client.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,287 @@ | |||||||
|  | from typing import TYPE_CHECKING | ||||||
|  | import asyncio | ||||||
|  | import NetUtils | ||||||
|  | import copy | ||||||
|  | from .Locations import grinch_locations, GrinchLocation | ||||||
|  | from .Items import ALL_ITEMS_TABLE, MISSION_ITEMS_TABLE, GADGETS_TABLE, KEYS_TABLE, GrinchItemData #, SLEIGH_PARTS_TABLE | ||||||
|  | import worlds._bizhawk as bizhawk | ||||||
|  | from worlds._bizhawk.client import BizHawkClient | ||||||
|  |  | ||||||
|  | if TYPE_CHECKING: | ||||||
|  |     from worlds._bizhawk.context import BizHawkClientContext | ||||||
|  |     from CommonClient import logger | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Stores received index of last item received in PS1 memory card save data | ||||||
|  | # By storing this index, it will remember the last item received and prevent item duplication loops | ||||||
|  | RECV_ITEM_ADDR = 0x010068 | ||||||
|  | RECV_ITEM_BITSIZE = 4 | ||||||
|  |  | ||||||
|  | # Maximum number of times we check if we are in demo mode or not | ||||||
|  | MAX_DEMO_MODE_CHECK = 30 | ||||||
|  |  | ||||||
|  | # List of Menu Map IDs | ||||||
|  | MENU_MAP_IDS: list[int] = [0x00, 0x02, 0x35, 0x36, 0x37] | ||||||
|  |  | ||||||
|  | class GrinchClient(BizHawkClient): | ||||||
|  |     game = "The Grinch" | ||||||
|  |     system = "PSX" | ||||||
|  |     patch_suffix = ".apgrinch" | ||||||
|  |     items_handling = 0b111 | ||||||
|  |     demo_mode_buffer = 0 | ||||||
|  |     last_map_location = -1 | ||||||
|  |     ingame_log = False | ||||||
|  |  | ||||||
|  |     def __init__(self): | ||||||
|  |         super().__init__() | ||||||
|  |         self.last_received_index = 0 | ||||||
|  |         self.loading_bios_msg = False | ||||||
|  |         self.loc_unlimited_eggs = False | ||||||
|  |  | ||||||
|  |     async def validate_rom(self, ctx: "BizHawkClientContext") -> bool: | ||||||
|  |         from CommonClient import logger | ||||||
|  |         # TODO Check the ROM data to see if it matches against bytes expected | ||||||
|  |         grinch_identifier_ram_address: int = 0x00928C | ||||||
|  |         bios_identifier_ram_address: int = 0x097F30 | ||||||
|  |         try: | ||||||
|  |             bytes_actual: bytes = (await bizhawk.read(ctx.bizhawk_ctx, [( | ||||||
|  |                 grinch_identifier_ram_address, 11, "MainRAM")]))[0] | ||||||
|  |  | ||||||
|  |             psx_rom_name = bytes_actual.decode("ascii") | ||||||
|  |             if psx_rom_name != "SLUS_011.97": | ||||||
|  |                 bios_bytes_check: bytes = (await bizhawk.read(ctx.bizhawk_ctx, [( | ||||||
|  |                     bios_identifier_ram_address, 24, "MainRAM")]))[0] | ||||||
|  |                 if "System ROM Version" in bios_bytes_check.decode("ascii"): | ||||||
|  |                     if not self.loading_bios_msg: | ||||||
|  |                         self.loading_bios_msg = True | ||||||
|  |                         logger.error("BIOS is currently loading. Will wait up to 5 seconds before retrying.") | ||||||
|  |                     return False | ||||||
|  |  | ||||||
|  |                 logger.error("Invalid rom detected. You are not playing Grinch USA Version.") | ||||||
|  |                 raise Exception("Invalid rom detected. You are not playing Grinch USA Version.") | ||||||
|  |         except Exception: | ||||||
|  |             return False | ||||||
|  |  | ||||||
|  |         ctx.game = self.game | ||||||
|  |         ctx.items_handling = self.items_handling | ||||||
|  |         ctx.want_slot_data = True | ||||||
|  |         ctx.watcher_timeout = 0.25 | ||||||
|  |         self.loading_bios_msg = False | ||||||
|  |  | ||||||
|  |         return True | ||||||
|  |  | ||||||
|  |     def on_package(self, ctx: "BizHawkClientContext", cmd: str, args: dict) -> None: | ||||||
|  |         from CommonClient import logger | ||||||
|  |         super().on_package(ctx, cmd, args) | ||||||
|  |         match cmd: | ||||||
|  |             case "Connected":  # On Connect | ||||||
|  |                 self.loc_unlimited_eggs = bool(ctx.slot_data["give_unlimited_eggs"]) | ||||||
|  |                 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.") | ||||||
|  |  | ||||||
|  |     async def set_auth(self, ctx: "BizHawkClientContext") -> None: | ||||||
|  |         await ctx.get_username() | ||||||
|  |  | ||||||
|  |     async def game_watcher(self, ctx: "BizHawkClientContext") -> None: | ||||||
|  |         from CommonClient import  logger | ||||||
|  |         #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: | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             if not await self.ingame_checker(ctx): | ||||||
|  |                 return | ||||||
|  |  | ||||||
|  |             await self.location_checker(ctx) | ||||||
|  |             await self.receiving_items_handler(ctx) | ||||||
|  |             await self.goal_checker(ctx) | ||||||
|  |             await self.option_handler(ctx) | ||||||
|  |             await self.constant_address_update(ctx) | ||||||
|  |  | ||||||
|  |         except bizhawk.RequestFailedError as ex: | ||||||
|  |             # 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)) | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |     async def location_checker(self, ctx: "BizHawkClientContext"): | ||||||
|  |         from CommonClient import logger | ||||||
|  |         # Update the AP Server to know what locations are not checked yet. | ||||||
|  |         local_locations_checked: list[int] = [] | ||||||
|  |         local_ap_locations: set[int] = copy.deepcopy(ctx.missing_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. | ||||||
|  |             # 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_ram_data = grinch_locations[grinch_loc_name] | ||||||
|  |  | ||||||
|  |             # Grinch ram data may have more than one address to update, so we are going to loop through all addresses in a location | ||||||
|  |             # We use a list here to keep track of all our checks. If they are all true, then and only then do we mark that location as checked. | ||||||
|  |             ram_checked_list: list[bool] = [] | ||||||
|  |             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 | ||||||
|  |                 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: | ||||||
|  |                     ram_checked_list.append((current_ram_address_value & (1 << addr_to_update.binary_bit_pos)) > 0) | ||||||
|  |                 else: | ||||||
|  |                     expected_int_value = addr_to_update.value | ||||||
|  |                     ram_checked_list.append(expected_int_value == current_ram_address_value) | ||||||
|  |             if all(ram_checked_list): | ||||||
|  |                 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) | ||||||
|  |         locations_sent_to_ap: set[int] = await ctx.check_locations(local_locations_checked) | ||||||
|  |         if len(locations_sent_to_ap) > 0: | ||||||
|  |             await self.remove_physical_items(ctx) | ||||||
|  |         ctx.locations_checked = set(local_locations_checked) | ||||||
|  |  | ||||||
|  |     async def receiving_items_handler(self, ctx: "BizHawkClientContext"): | ||||||
|  |         # Len will give us the size of the items received list & we will track that against how many items we received already | ||||||
|  |         # If the list says that we have 3 items and we already received items, we will ignore and continue. | ||||||
|  |         # Otherwise, we will get the new items and give them to the player. | ||||||
|  |  | ||||||
|  |         self.last_received_index = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( | ||||||
|  |             RECV_ITEM_ADDR, RECV_ITEM_BITSIZE, "MainRAM")]))[0], "little") | ||||||
|  |         if len(ctx.items_received) == self.last_received_index: | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         # Ensures we only get the new items that we want to give the player | ||||||
|  |         new_items_only = ctx.items_received[self.last_received_index:] | ||||||
|  |  | ||||||
|  |         for item_received in new_items_only: | ||||||
|  |             local_item = ctx.item_names.lookup_in_game(item_received.item) | ||||||
|  |             grinch_item_ram_data = ALL_ITEMS_TABLE[local_item] | ||||||
|  |  | ||||||
|  |             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 | ||||||
|  |                 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: | ||||||
|  |                     current_ram_address_value = (current_ram_address_value | (1 << addr_to_update.binary_bit_pos)) | ||||||
|  |                 elif addr_to_update.update_existing_value: | ||||||
|  |                     # Grabs minimum value of a list of numbers and makes sure it does not go above max count possible | ||||||
|  |                     current_ram_address_value += addr_to_update.value | ||||||
|  |                     current_ram_address_value = min(current_ram_address_value, addr_to_update.max_count) | ||||||
|  |                 else: | ||||||
|  |                     current_ram_address_value = addr_to_update.value | ||||||
|  |  | ||||||
|  |                 # 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) | ||||||
|  |                 # 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 | ||||||
|  |             await self.update_and_validate_address(ctx, RECV_ITEM_ADDR, self.last_received_index, RECV_ITEM_BITSIZE) | ||||||
|  |  | ||||||
|  |     async def goal_checker(self, ctx: "BizHawkClientContext"): | ||||||
|  |         if not ctx.finished_game: | ||||||
|  |             goal_loc = grinch_locations["Neutralizing Santa"] | ||||||
|  |             goal_ram_address = goal_loc.update_ram_addr[0] | ||||||
|  |             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") | ||||||
|  |             if (current_ram_address_value & (1 << goal_ram_address.binary_bit_pos)) > 0: | ||||||
|  |                 ctx.finished_game = True | ||||||
|  |                 await ctx.send_msgs([{ | ||||||
|  |                     "cmd": "StatusUpdate", | ||||||
|  |                     "status": NetUtils.ClientStatus.CLIENT_GOAL, | ||||||
|  |                 }]) | ||||||
|  |  | ||||||
|  |     # 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"): | ||||||
|  |         list_recv_itemids: list[int] = [netItem.item for netItem in ctx.items_received] | ||||||
|  |         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_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) | ||||||
|  |  | ||||||
|  |         # Setting Who Lake Mission Count back to 0 to prevent warping after completing 3 missions | ||||||
|  |         await self.update_and_validate_address(ctx,0x0100F0, 0, 4) | ||||||
|  |  | ||||||
|  |         for (item_name, item_data) in items_to_check.items(): | ||||||
|  |             # 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: | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             # This assumes we don't have the item so we must set all the data to 0 | ||||||
|  |             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 | ||||||
|  |                 if is_binary: | ||||||
|  |                     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) | ||||||
|  |                     await self.update_and_validate_address(ctx, addr_to_update.ram_address, current_bin_value, 1) | ||||||
|  |                 else: | ||||||
|  |                     await self.update_and_validate_address(ctx, addr_to_update.ram_address, 0, 1) | ||||||
|  |  | ||||||
|  |     # Removes the regional access until you actually received it from AP. | ||||||
|  |     async def constant_address_update(self, ctx: "BizHawkClientContext"): | ||||||
|  |         list_recv_itemids: list[int] = [netItem.item for netItem in ctx.items_received] | ||||||
|  |         items_to_check: dict[str, GrinchItemData] = {**KEYS_TABLE, **MISSION_ITEMS_TABLE} | ||||||
|  |  | ||||||
|  |         for (item_name, item_data) in items_to_check.items(): | ||||||
|  |             # 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: | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             # This assumes we don't have the item so we must set all the data to 0 | ||||||
|  |             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 | ||||||
|  |                 if is_binary: | ||||||
|  |                     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) | ||||||
|  |                     await self.update_and_validate_address(ctx, addr_to_update.ram_address, current_bin_value, 1) | ||||||
|  |                 else: | ||||||
|  |                     await self.update_and_validate_address(ctx, addr_to_update.ram_address, 0, 1) | ||||||
|  |  | ||||||
|  |     async def ingame_checker(self, ctx: "BizHawkClientContext"): | ||||||
|  |         from CommonClient import logger | ||||||
|  |  | ||||||
|  |         ingame_map_id = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( | ||||||
|  |             0x010000, 1, "MainRAM")]))[0], "little") | ||||||
|  |  | ||||||
|  |         #If not in game or at a menu, or loading the publisher logos | ||||||
|  |         if ingame_map_id <= 0x04 or ingame_map_id >= 0x35: | ||||||
|  |             return False | ||||||
|  |  | ||||||
|  |         #If grinch has changed maps | ||||||
|  |         if not ingame_map_id == self.last_map_location: | ||||||
|  |             # If the last "map" we were on was a menu or a publisher logo | ||||||
|  |             if self.last_map_location in MENU_MAP_IDS: | ||||||
|  |                 # Reset our demo mode checker just in case the game is in demo mode. | ||||||
|  |                 self.demo_mode_buffer = 0 | ||||||
|  |                 self.ingame_log = False | ||||||
|  |                 return False | ||||||
|  |  | ||||||
|  |             # Update the previous map we were on to be the current map. | ||||||
|  |             self.last_map_location = ingame_map_id | ||||||
|  |  | ||||||
|  |         # Use this as a delayed check to make sure we are in game | ||||||
|  |         if not self.demo_mode_buffer == MAX_DEMO_MODE_CHECK: | ||||||
|  |             await asyncio.sleep(0.1) | ||||||
|  |             self.demo_mode_buffer += 1 | ||||||
|  |             return False | ||||||
|  |  | ||||||
|  |         demo_mode = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [( | ||||||
|  |             0x01008A, 1, "MainRAM")]))[0], "little") | ||||||
|  |         if demo_mode == 1: | ||||||
|  |             return False | ||||||
|  |  | ||||||
|  |         if not self.ingame_log: | ||||||
|  |             logger.info("You can now start sending locations from the Grinch!") | ||||||
|  |             self.ingame_log = True | ||||||
|  |         return True | ||||||
|  |  | ||||||
|  |     async def option_handler(self, ctx: "BizHawkClientContext"): | ||||||
|  |         if self.loc_unlimited_eggs: | ||||||
|  |             max_eggs: int = 200 | ||||||
|  |             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): | ||||||
|  |         await bizhawk.write(ctx.bizhawk_ctx, [(address_to_validate, expected_value.to_bytes(byte_size, "little"), "MainRAM")]) | ||||||
|  |         current_value = int.from_bytes((await bizhawk.read(ctx.bizhawk_ctx, [(address_to_validate, byte_size, "MainRAM")]))[0], "little") | ||||||
|  |         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. | ||||||
|  |                 return | ||||||
|  |             raise Exception("Unable to update address as expected. Address: "+ str(address_to_validate)+"; Expected Value: "+str(expected_value)) | ||||||
							
								
								
									
										242
									
								
								worlds/grinch/Items.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										242
									
								
								worlds/grinch/Items.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,242 @@ | |||||||
|  | from typing import NamedTuple, Optional | ||||||
|  |  | ||||||
|  | from .RamHandler import GrinchRamData | ||||||
|  | from BaseClasses import Item | ||||||
|  | from BaseClasses import ItemClassification as IC #IC can be any name, saves having to type the whole word in code | ||||||
|  |  | ||||||
|  | class GrinchItemData(NamedTuple): | ||||||
|  |     item_group: str #arbituary that can be whatever it can be, basically the field/property for item groups | ||||||
|  |     id: Optional[int] | ||||||
|  |     classification: IC | ||||||
|  |     update_ram_addr: list[GrinchRamData] | ||||||
|  |     second_item_group: Optional[str] = None | ||||||
|  |  | ||||||
|  | class GrinchItem(Item): | ||||||
|  |     game: str = "The Grinch" | ||||||
|  |  | ||||||
|  |     #Tells server what item id it is | ||||||
|  |     @staticmethod | ||||||
|  |     def get_apid(id: int): | ||||||
|  |         #If you give me an input id, I will return the Grinch equivalent server/ap id | ||||||
|  |         base_id: int = 42069 | ||||||
|  |         return base_id + id if id is not None else None | ||||||
|  |  | ||||||
|  |     def __init__(self, name: str, player: int, data: GrinchItemData): | ||||||
|  |         super(GrinchItem, self).__init__(name,data.classification, GrinchItem.get_apid(data.id), player) | ||||||
|  |  | ||||||
|  |         self.type = data.item_group | ||||||
|  |         self.item_id = data.id | ||||||
|  |  | ||||||
|  | #allows hinting of items via category | ||||||
|  | def get_item_names_per_category() -> dict[str, set[str]]: | ||||||
|  |     categories: dict[str, set[str]] = {} | ||||||
|  |  | ||||||
|  |     for name, data in ALL_ITEMS_TABLE.items(): | ||||||
|  |         categories.setdefault(data.item_group, set()).add(name) | ||||||
|  |  | ||||||
|  |     return categories | ||||||
|  |  | ||||||
|  | #Gadgets | ||||||
|  | #All gadgets require at least 4 different blueprints to be unlocked in the computer in Mount Crumpit. | ||||||
|  | GADGETS_TABLE: dict[str, GrinchItemData] = { | ||||||
|  |     "Binoculars": GrinchItemData("Gadgets", 100, IC.useful, | ||||||
|  |         [GrinchRamData(0x0102B6, value=0x40), GrinchRamData(0x0102B7, value=0x41), | ||||||
|  |         GrinchRamData(0x0102B8, value=0x44), GrinchRamData(0x0102B9, value=0x45), | ||||||
|  |         # GrinchRamData(0x0100BC, binary_bit_pos=0) | ||||||
|  |          ]), | ||||||
|  |     "Rotten Egg Launcher": GrinchItemData("Gadgets", 101, IC.progression, | ||||||
|  |         [GrinchRamData(0x0102BA, value=0x40), GrinchRamData(0x0102BB, value=0x41), | ||||||
|  |         GrinchRamData(0x0102BC, value=0x44), GrinchRamData(0x0102BD, value=0x45), | ||||||
|  |         # GrinchRamData(0x0100BC, binary_bit_pos=1) | ||||||
|  |          ]), | ||||||
|  |     "Rocket Spring": GrinchItemData("Gadgets", 102, IC.progression, | ||||||
|  |         [GrinchRamData(0x0102BE, value=0x40), GrinchRamData(0x0102BF, value=0x41), | ||||||
|  |         GrinchRamData(0x0102C0, value=0x42), GrinchRamData(0x0102C1, value=0x44), | ||||||
|  |         GrinchRamData(0x0102C2, value=0x45), GrinchRamData(0x0102C3, value=0x46), | ||||||
|  |         GrinchRamData(0x0102C4, value=0x48), GrinchRamData(0x0102C5, value=0x49), | ||||||
|  |         GrinchRamData(0x0102C6, value=0x4A), | ||||||
|  |          # GrinchRamData(0x0100BC, binary_bit_pos=2) | ||||||
|  |          ]), | ||||||
|  |     "Slime Shooter": GrinchItemData("Gadgets", 103, IC.progression, | ||||||
|  |         [GrinchRamData(0x0102C7, value=0x40), GrinchRamData(0x0102C8, value=0x41), | ||||||
|  |         GrinchRamData(0x0102C9, value=0x42), GrinchRamData(0x0102CA, value=0x44), | ||||||
|  |         GrinchRamData(0x0102CB, value=0x45), GrinchRamData(0x0102CC, value=0x46), | ||||||
|  |         GrinchRamData(0x0102CD, value=0x48), GrinchRamData(0x0102CE, value=0x49), | ||||||
|  |         GrinchRamData(0x0102CF, value=0x4A), | ||||||
|  |          # GrinchRamData(0x0100BC, binary_bit_pos=3) | ||||||
|  |          ]), | ||||||
|  |     "Octopus Climbing Device": GrinchItemData("Gadgets", 104, IC.progression, | ||||||
|  |         [GrinchRamData(0x0102D0, value=0x40), GrinchRamData(0x0102D1, value=0x41), | ||||||
|  |         GrinchRamData(0x0102D2, value=0x42), GrinchRamData(0x0102D3, value=0x44), | ||||||
|  |         GrinchRamData(0x0102D4, value=0x45), GrinchRamData(0x0102D5, value=0x46), | ||||||
|  |         GrinchRamData(0x0102D6, value=0x48), GrinchRamData(0x0102D7, value=0x49), | ||||||
|  |         GrinchRamData(0x0102D8, value=0x4A), | ||||||
|  |          # GrinchRamData(0x0100BC, binary_bit_pos=4) | ||||||
|  |          ]), | ||||||
|  |     "Marine Mobile": GrinchItemData("Gadgets", 105, IC.progression, | ||||||
|  |         [GrinchRamData(0x0102D9, value=0x40), GrinchRamData(0x0102DA, value=0x41), | ||||||
|  |         GrinchRamData(0x0102DB, value=0x42), GrinchRamData(0x0102DC, value=0x43), | ||||||
|  |         GrinchRamData(0x0102DD, value=0x44), GrinchRamData(0x0102DE, value=0x45), | ||||||
|  |         GrinchRamData(0x0102DF, value=0x46), GrinchRamData(0x0102E0, value=0x47), | ||||||
|  |         GrinchRamData(0x0102E1, value=0x48), GrinchRamData(0x0102E2, value=0x49), | ||||||
|  |         GrinchRamData(0x0102E3, value=0x4A), GrinchRamData(0x0102E4, value=0x4B), | ||||||
|  |         GrinchRamData(0x0102E5, value=0x4C), GrinchRamData(0x0102E6, value=0x4D), | ||||||
|  |         GrinchRamData(0x0102E7, value=0x4E), GrinchRamData(0x0102E8, value=0x4F), | ||||||
|  |         # GrinchRamData(0x0100BC, binary_bit_pos=5) | ||||||
|  |          ]), | ||||||
|  |     "Grinch Copter": GrinchItemData("Gadgets", 106, IC.progression, | ||||||
|  |         [GrinchRamData(0x0102E9, value=0x40), GrinchRamData(0x0102EA, value=0x41), | ||||||
|  |         GrinchRamData(0x0102EB, value=0x42), GrinchRamData(0x0102EC, value=0x43), | ||||||
|  |         GrinchRamData(0x0102ED, value=0x44), GrinchRamData(0x0102EE, value=0x45), | ||||||
|  |         GrinchRamData(0x0102EF, value=0x46), GrinchRamData(0x0102F0, value=0x47), | ||||||
|  |         GrinchRamData(0x0102F1, value=0x48), GrinchRamData(0x0102F2, value=0x49), | ||||||
|  |         GrinchRamData(0x0102F3, value=0x4A), GrinchRamData(0x0102F4, value=0x4B), | ||||||
|  |         GrinchRamData(0x0102F5, value=0x4C), GrinchRamData(0x0102F6, value=0x4D), | ||||||
|  |         GrinchRamData(0x0102F7, value=0x4E), GrinchRamData(0x0102F8, value=0x4F), | ||||||
|  |         # GrinchRamData(0x0100BC, binary_bit_pos=6) | ||||||
|  |     ]) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #Mission Specific Items | ||||||
|  | MISSION_ITEMS_TABLE: dict[str, GrinchItemData] = { | ||||||
|  |     "Who Cloak": GrinchItemData("Mission Specific Items", 200, IC.progression, | ||||||
|  |         [GrinchRamData(0x0101F9, binary_bit_pos=0)], second_item_group="Useful Items"), | ||||||
|  |     "Painting Bucket": GrinchItemData("Mission Specific Items", 201, IC.progression_deprioritized, | ||||||
|  |         [GrinchRamData(0x0101F9, binary_bit_pos=1)], second_item_group="Useful Items"), | ||||||
|  |     "Scissors": GrinchItemData("Mission Specific Items", 202, IC.progression_deprioritized, | ||||||
|  |         [GrinchRamData(0x0101F9, binary_bit_pos=6), GrinchRamData(0x0100C2, binary_bit_pos=1)], | ||||||
|  |         second_item_group="Useful Items"), | ||||||
|  |     "Glue Bucket": GrinchItemData("Mission Specific Items", 203, IC.progression_deprioritized, | ||||||
|  |         [GrinchRamData(0x0101F9, binary_bit_pos=4)], second_item_group="Useful Items"), | ||||||
|  |     "Cable Car Access Card": GrinchItemData("Mission Specific Items", 204, IC.progression, | ||||||
|  |         [GrinchRamData(0x0101F9, binary_bit_pos=5)], second_item_group="Useful Items"), | ||||||
|  |     "Drill": GrinchItemData("Mission Specific Items", 205, IC.progression_deprioritized, | ||||||
|  |         [GrinchRamData(0x0101FA, binary_bit_pos=2)], second_item_group="Useful Items"), | ||||||
|  |     "Rope": GrinchItemData("Mission Specific Items", 206, IC.progression_deprioritized, | ||||||
|  |         [GrinchRamData(0x0101FA, binary_bit_pos=1)], second_item_group="Useful Items"), | ||||||
|  |     "Hook": GrinchItemData("Mission Specific Items", 207, IC.progression_deprioritized, | ||||||
|  |         [GrinchRamData(0x0101FA, binary_bit_pos=0)], second_item_group="Useful Items"), | ||||||
|  |     "Sculpting Tools": GrinchItemData("Mission Specific Items", 208, IC.progression_deprioritized, | ||||||
|  |         [GrinchRamData(0x0101F9, binary_bit_pos=2)], second_item_group="Useful Items"), | ||||||
|  |     "Hammer": GrinchItemData("Mission Specific Items", 209, IC.progression_deprioritized, | ||||||
|  |         [GrinchRamData(0x0101F9, binary_bit_pos=3)], second_item_group="Useful Items"), | ||||||
|  |     "Scout Clothes": GrinchItemData("Mission Specific Items", 210, IC.progression, | ||||||
|  |         [GrinchRamData(0x0101F9, binary_bit_pos=7)], second_item_group="Useful Items") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #Sleigh Parts | ||||||
|  | # SLEIGH_PARTS_TABLE: dict[str, GrinchItemData] = { | ||||||
|  | #     "Exhaust Pipes": GrinchItemData("Sleigh Parts", 300, IC.progression_skip_balancing, | ||||||
|  | #         [GrinchRamData(0x0101FB, binary_bit_pos=2)]), | ||||||
|  | #     "GPS": GrinchItemData("Sleigh Parts", 301, IC.useful, | ||||||
|  | #         [GrinchRamData(0x0101FB, binary_bit_pos=5)]), | ||||||
|  | #     "Tires": GrinchItemData("Sleigh Parts", 302, IC.progression_skip_balancing, | ||||||
|  | #         [GrinchRamData(0x0101FB, binary_bit_pos=4)]), | ||||||
|  | #     "Skis": GrinchItemData("Sleigh Parts", 303, IC.progression_skip_balancing, | ||||||
|  | #         [GrinchRamData(0x0101FB, binary_bit_pos=3)]), | ||||||
|  | #     "Twin-End Tuba": GrinchItemData("Sleigh Parts", 304, IC.progression_skip_balancing, | ||||||
|  | #         [GrinchRamData(0x0101FB, binary_bit_pos=6)]) | ||||||
|  | # } | ||||||
|  |  | ||||||
|  | #Access Keys | ||||||
|  | KEYS_TABLE: dict[str, GrinchItemData] = { | ||||||
|  |     # "Whoville Vacuum Access": GrinchItemData("Vacuum Access", 400, IC.progression, | ||||||
|  |     #     [GrinchRamData()]), | ||||||
|  |     "Who Forest Vacuum Access": GrinchItemData("Vacuum Access", 401, IC.progression, | ||||||
|  |         [GrinchRamData(0x0100AA, binary_bit_pos=2)]), | ||||||
|  |     "Who Dump Vacuum Access": GrinchItemData("Vacuum Access", 402, IC.progression, | ||||||
|  |         [GrinchRamData(0x0100AA, binary_bit_pos=3)]), | ||||||
|  |     "Who Lake Vacuum Access": GrinchItemData("Vacuum Access", 403, IC.progression, | ||||||
|  |         [GrinchRamData(0x0100AA, binary_bit_pos=4)]), | ||||||
|  |     # "Progressive Vacuum Access": GrinchItemData("Vacuum Access", 404, IC.progression, | ||||||
|  |     #     [GrinchRamData()]), | ||||||
|  |     # "Spin N' Win Door Unlock": GrinchItemData("Supadow Door Unlocks", 405, IC.progression, | ||||||
|  |     #     [GrinchRamData()]), | ||||||
|  |     # "Dankamania Door Unlock": GrinchItemData("Supadow Door Unlocks", 406, IC.progression, | ||||||
|  |     #     [GrinchRamData()]), | ||||||
|  |     # "The Copter Race Contest Door Unlock": GrinchItemData("Supadow Door Unlocks", 407, IC.progression, | ||||||
|  |     #     [GrinchRamData()]), | ||||||
|  |     # "Progressive Supadow Door Unlock": GrinchItemData("Supadow Door Unlocks", 408, IC.progression, | ||||||
|  |     #     [GrinchRamData()]), | ||||||
|  |     # "Bike Race Access": GrinchItemData("Supadow Door Unlocks", 409, IC.progression, | ||||||
|  |     #     [GrinchRamData()]) | ||||||
|  |     "Sleigh Room Key": GrinchItemData("Sleigh Room", 410, IC.progression, | ||||||
|  |         [GrinchRamData(0x010200, binary_bit_pos=6), GrinchRamData(0x0100AA, binary_bit_pos=5)]) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #Misc Items | ||||||
|  | MISC_ITEMS_TABLE: dict[str, GrinchItemData] = { | ||||||
|  |     # 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, | ||||||
|  |     #     [GrinchRamData(0x0E8FDC, value=120)]), | ||||||
|  |     "5 Rotten Eggs": GrinchItemData("Rotten Egg Bundles", 502, IC.filler, | ||||||
|  |         [GrinchRamData(0x010058, value=5, update_existing_value=True, max_count=200, bit_size=2)]), | ||||||
|  |     "10 Rotten Eggs": GrinchItemData("Rotten Egg Bundles", 503, IC.filler, | ||||||
|  |         [GrinchRamData(0x010058, value=10, update_existing_value=True, max_count=200, bit_size=2)]), | ||||||
|  |     "20 Rotten Eggs": GrinchItemData("Rotten Egg Bundles", 504, IC.filler, | ||||||
|  |         [GrinchRamData(0x010058, value=20, update_existing_value=True, max_count=200, bit_size=2)]) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | USEFUL_IC_TABLE: dict[str, GrinchItemData] = { | ||||||
|  |     "Heart of Stone": GrinchItemData("Health Items", 501, IC.useful, | ||||||
|  |         [GrinchRamData(0x0100ED, value=1, update_existing_value=True, max_count=4)]) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #Traps | ||||||
|  | TRAPS_TABLE: dict[str, GrinchItemData] = { | ||||||
|  | # alias to Ice Trap for traplink | ||||||
|  |     # "Freeze Trap": GrinchItemData("Traps", 600, IC.trap, [GrinchRamData()]), | ||||||
|  |     # "Bee Trap": GrinchItemData("Traps", 601, IC.trap, [GrinchRamData()]), | ||||||
|  |     # "Electrocution Trap": GrinchItemData("Traps", 602, IC.trap, [GrinchRamData()]), | ||||||
|  | # alias to Slowness Trap for traplink | ||||||
|  |     # "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 | ||||||
|  | #     "Damage Trap": GrinchItemData("Traps", 604, IC.trap, [GrinchRamData(0x0E8FDC, value=-20, update_existing_value=True)]), | ||||||
|  |     "Depletion Trap": GrinchItemData("Traps", 605, IC.trap, [GrinchRamData(0x010058, value=0, bit_size=2)]), | ||||||
|  |     "Dump it to Crumpit": GrinchItemData("Traps", 606, IC.trap, #Alias to Home Trap for traplink | ||||||
|  |         [GrinchRamData(0x010000, value=0x05), GrinchRamData(0x08FB94, value=1)]), | ||||||
|  | #alias to Spring Trap for traplink | ||||||
|  |     # "Rocket Spring Trap": GrinchItemData("Traps", 607, IC.trap, [GrinchRamData()]), | ||||||
|  | #alias to Home Trap for traplink | ||||||
|  |     "Who sent me back?": GrinchItemData("Traps", 608, IC.trap, [GrinchRamData(0x08FB94, value=1)]), | ||||||
|  |     # "Cutscene Trap": GrinchItemData("Traps", 609, IC.trap, [GrinchRamData()]), | ||||||
|  |     # "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)]) | ||||||
|  |     # "Child Trap": GrinchItemData("Traps", 612, IC.trap,[GrinchRamData()]) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #Movesets | ||||||
|  | # MOVES_TABLE: dict[str, GrinchItemData] = { | ||||||
|  | #     "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)]), | ||||||
|  | #     "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)]), | ||||||
|  | #     "Tip Toe": GrinchItemData("Movesets", 704, IC.progression, [GrinchRamData(0x0100BB, binary_bit_pos=5)]) | ||||||
|  | # } | ||||||
|  | #Double star combines all dictionaries from each individual list together | ||||||
|  | ALL_ITEMS_TABLE: dict[str, GrinchItemData] = { | ||||||
|  |     **GADGETS_TABLE, | ||||||
|  |     **MISSION_ITEMS_TABLE, | ||||||
|  |     **KEYS_TABLE, | ||||||
|  |     **MISC_ITEMS_TABLE, | ||||||
|  |     **TRAPS_TABLE, | ||||||
|  |     **USEFUL_IC_TABLE, | ||||||
|  |     # **SLEIGH_PARTS_TABLE, | ||||||
|  |     # **MOVES_TABLE, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # Psuedocoding traplink table | ||||||
|  | # 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"] | ||||||
|  | # 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"] | ||||||
|  | # SPRING_TRAP_EQUIV = ["Eject Ability", "Hiccup Trap", "Jump Trap", "Jumping Jacks Trap", "Whoops! Trap"] | ||||||
|  | # HOME_TRAP_EQUIV = ["Blue Balls Curse", "Instant Death Trap"] | ||||||
|  | # SLOWNESS_TRAP_EQUIV = ["Iron Boots Trap", "Slow Trap", "Sticky Floor Trap"] | ||||||
|  | # CUTSCENE_TRAP_EQUIV = ["Phone Trap"] | ||||||
|  | # ELEC_TRAP_EQUIV = [] | ||||||
|  |  | ||||||
|  | def grinch_items_to_id() -> dict[str, int]: | ||||||
|  |     item_mappings: dict[str, int] = {} | ||||||
|  |     for ItemName, ItemData in ALL_ITEMS_TABLE.items(): | ||||||
|  |         item_mappings.update({ItemName: GrinchItem.get_apid(ItemData.id)}) | ||||||
|  |     return item_mappings | ||||||
							
								
								
									
										191
									
								
								worlds/grinch/Locations.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										191
									
								
								worlds/grinch/Locations.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,191 @@ | |||||||
|  | from typing import NamedTuple, Optional | ||||||
|  |  | ||||||
|  | from .RamHandler import GrinchRamData | ||||||
|  | from BaseClasses import Location, Region | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GrinchLocationData(NamedTuple): | ||||||
|  |     region: str | ||||||
|  |     location_group: str | ||||||
|  |     id: Optional[int] | ||||||
|  |     update_ram_addr: list[GrinchRamData] | ||||||
|  |     reset_addr: Optional[list[GrinchRamData]] = None # Addresses to update once we find the item | ||||||
|  |  | ||||||
|  | class GrinchLocation(Location): | ||||||
|  |     game: str = "The Grinch" | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def get_apid(id: int): | ||||||
|  |         base_id: int = 42069 | ||||||
|  |         return base_id + id if id is not None else None | ||||||
|  |  | ||||||
|  |     def __init__(self, player: int, name: str, parent: Region, data: GrinchLocationData): | ||||||
|  |         address = None if data.id is None else GrinchLocation.get_apid(data.id) | ||||||
|  |         super(GrinchLocation, self).__init__(player, name, address=address, parent=parent) | ||||||
|  |  | ||||||
|  |         self.code = data.id | ||||||
|  |         self.region = data.region | ||||||
|  |         self.type = data.location_group | ||||||
|  |         self.address = self.address | ||||||
|  |  | ||||||
|  | grinch_locations = { | ||||||
|  | #Going to use current map id as indicator whether or not you visited a location | ||||||
|  | #Visitsanity | ||||||
|  |         "Enter Whoville": GrinchLocationData("Whoville", "Visitsanity", 100, [GrinchRamData(0x010000, value=0x07)]), | ||||||
|  |         "Enter the Post Office": GrinchLocationData("Post Office", "Visitsanity", 101, [GrinchRamData(0x010000, value=0x0A)]), | ||||||
|  |         "Enter the Town Hall": GrinchLocationData("City Hall", "Visitsanity", 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)]), | ||||||
|  |         "Enter Who Forest": GrinchLocationData("Who Forest", "Visitsanity", 104, [GrinchRamData(0x010000, value=0x0B)]), | ||||||
|  |         "Enter the Ski Resort": GrinchLocationData("Ski Resort", "Visitsanity", 105, [GrinchRamData(0x010000, value=0x0C)]), | ||||||
|  |         "Enter the Civic Center": GrinchLocationData("Civic Center", "Visitsanity", 106, [GrinchRamData(0x010000, value=0x0D)]), | ||||||
|  |         "Enter Who Dump": GrinchLocationData("Who Dump", "Visitsanity", 107, [GrinchRamData(0x010000, value=0x0E)]), | ||||||
|  |         "Enter the Minefield": GrinchLocationData("Minefield", "Visitsanity", 108, [GrinchRamData(0x010000, value=0x11)]), | ||||||
|  |         "Enter the Power Plant": GrinchLocationData("Power Plant", "Visitsanity", 109, [GrinchRamData(0x010000, value=0x10)]), | ||||||
|  |         "Enter the Generator Building": GrinchLocationData("Generator Building", "Visitsanity", 110, [GrinchRamData(0x010000, value=0x0F)]), | ||||||
|  |         "Enter Who Lake": GrinchLocationData("Who Lake", "Visitsanity", 111, [GrinchRamData(0x010000, value=0x12)]), | ||||||
|  |         "Enter the Submarine World": GrinchLocationData("Submarine World", "Visitsanity", 112, [GrinchRamData(0x010000, value=0x17)]), | ||||||
|  |         "Enter the Scout's Hut": GrinchLocationData("Scout's Hut", "Visitsanity", 113, [GrinchRamData(0x010000, value=0x13)]), | ||||||
|  |         "Enter the North Shore": GrinchLocationData("North Shore", "Visitsanity", 114, [GrinchRamData(0x010000, value=0x14)]), | ||||||
|  |         "Enter the Mayor's Villa": GrinchLocationData("Mayor's Villa", "Visitsanity", 115, [GrinchRamData(0x010000, value=0x16)]), | ||||||
|  | #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 | ||||||
|  | #Whoville Missions | ||||||
|  |         "Shuffling The Mail": GrinchLocationData("Post Office", "Whoville Missions", 201, [GrinchRamData(0x0100BE, binary_bit_pos=0)]), | ||||||
|  |         "Smashing Snowmen": GrinchLocationData("Whoville", "Whoville Missions", 200, [GrinchRamData(0x0100C5, value=10)]), | ||||||
|  |         "Painting The Mayor's Posters": GrinchLocationData("Whoville", "Whoville Missions", 202, [GrinchRamData(0x0100C6, value=10)]), | ||||||
|  |         "Launching Eggs Into Houses": GrinchLocationData("Whoville", "Whoville Missions", 203, [GrinchRamData(0x0100C7, value=10)]), | ||||||
|  |         "Modifying The Mayor's Statue": GrinchLocationData("City Hall", "Whoville Missions", 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)]), | ||||||
|  |         "Squashing All Gifts in Whoville": GrinchLocationData("Whoville", "Whoville Missions", 206, [GrinchRamData(0x01005C, value=500, bit_size=2)]), | ||||||
|  | #Who Forest Missions | ||||||
|  |         "Making Xmas Trees Droop": GrinchLocationData("Who Forest", "Who Forest Missions", 300, [GrinchRamData(0x0100C8, value=10)]), | ||||||
|  |         "Sabotaging Snow Cannon With Glue": GrinchLocationData("Who Forest", "Who Forest Missions", 301, [GrinchRamData(0x0100BE, binary_bit_pos=3)]), | ||||||
|  |         "Putting Beehives In Cabins": GrinchLocationData("Who Forest", "Who Forest Missions", 302, [GrinchRamData(0x0100CA, value=10)]), | ||||||
|  |         "Sliming The Mayor's Skis": GrinchLocationData("Ski Resort", "Who Forest Missions", 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)]), | ||||||
|  |         "Squashing All Gifts in Who Forest": GrinchLocationData("Who Forest", "Who Forest Missions", 305, [GrinchRamData(0x01005E, value=750, bit_size=2)]), | ||||||
|  | #Who Dump Missions | ||||||
|  |         "Stealing Food From Birds": GrinchLocationData("Who Dump", "Who Dump Missions", 400, [GrinchRamData(0x0100CB, value=10)]), | ||||||
|  |         "Feeding The Computer With Robot Parts": GrinchLocationData("Who Dump", "Who Dump Missions", 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)]), | ||||||
|  |         "Conducting The Stinky Gas To Who-Bris' Shack": GrinchLocationData("Who Dump", "Who Dump Missions", 403, [GrinchRamData(0x0100BE, binary_bit_pos=7)]), | ||||||
|  |         "Shaving Who Dump Guardian": GrinchLocationData("Minefield", "Who Dump Missions", 404, [GrinchRamData(0x0100BF, binary_bit_pos=0)]), | ||||||
|  |         "Short-Circuiting Power-Plant": GrinchLocationData("Generator Building", "Who Dump Missions", 405, [GrinchRamData(0x0100BF, binary_bit_pos=1)]), | ||||||
|  |         "Squashing All Gifts in Who Dump": GrinchLocationData("Who Dump", "Who Dump Missions", 406, [GrinchRamData(0x010060, value=750, bit_size=2)]), | ||||||
|  | #Who Lake Missions | ||||||
|  |         "Putting Thistles In Shorts": GrinchLocationData("Who Lake", "Who Lake Missions", 500, [GrinchRamData(0x0100E5, value=10)]), | ||||||
|  |         "Sabotaging The Tents": GrinchLocationData("Who Lake", "Who Lake Missions", 501, [GrinchRamData(0x0100E6, value=10)]), | ||||||
|  |         "Drilling Holes In Canoes": GrinchLocationData("North Shore", "Who Lake Missions", 502, [GrinchRamData(0x0100EE, value=10)]), | ||||||
|  |         "Modifying The Marine Mobile": GrinchLocationData("Submarine World", "Who Lake Missions", 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)]), | ||||||
|  |         "Squashing All Gifts in Who Lake": GrinchLocationData("Who Lake", "Who Lake Missions", 505, [GrinchRamData(0x010062, value=1000, bit_size=2)]), | ||||||
|  | #Need to find binary values for individual blueprints, but all ram addresses are found | ||||||
|  | #Blueprints | ||||||
|  | #Binoculars Blueprints | ||||||
|  |         "Binoculars Blueprint - Post Office Roof": GrinchLocationData("Whoville", "Binocular 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)]), | ||||||
|  |         "Binoculars Blueprint - City Hall Library - Front Side": GrinchLocationData("City Hall", "Binocular 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)]), | ||||||
|  | #Rotten Egg Launcher Blueprints | ||||||
|  |         "REL Blueprint - Outside City Hall": GrinchLocationData("Whoville", "Rotten Egg Launcher 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)]), | ||||||
|  |         "REL Blueprint - Post Office - Inside Silver Room": GrinchLocationData("Post Office", "Rotten Egg Launcher 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)]), | ||||||
|  | #Rocket Spring Blueprints | ||||||
|  |         "RS Blueprint - Behind Vacuum": GrinchLocationData("Who Forest", "Rocket Spring 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)]), | ||||||
|  |         "RS Blueprint - Near Tree House on Ground": GrinchLocationData("Who Forest", "Rocket Spring 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)]), | ||||||
|  |         "RS Blueprint - Near Who Snowball in Cave": GrinchLocationData("Who Forest", "Rocket Spring 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)]), | ||||||
|  |         "RS Blueprint - Branch Platform Near Beast": GrinchLocationData("Who Forest", "Rocket Spring 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)]), | ||||||
|  |         "RS Blueprint - On Tree House": GrinchLocationData("Who Forest", "Rocket Spring Blueprints", 809, [GrinchRamData(0x010243, binary_bit_pos=5)]), | ||||||
|  | #Slime Shooter Blueprints | ||||||
|  |         "SS Blueprint - Branch Platform Elevated House": GrinchLocationData("Who Forest", "Slime Shooter 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)]), | ||||||
|  |         "SS Blueprint - House near Civic Center Cave": GrinchLocationData("Who Forest", "Slime Shooter 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)]), | ||||||
|  |         "SS Blueprint - House across from Tree House": GrinchLocationData("Who Forest", "Slime Shooter 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)]), | ||||||
|  |         "SS Blueprint - 2nd House near entrance left side": GrinchLocationData("Who Forest", "Slime Shooter 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)]), | ||||||
|  |         "SS Blueprint - House near entrance": GrinchLocationData("Who Forest", "Slime Shooter Blueprints", 908, [GrinchRamData(0x010244, binary_bit_pos=0)]), | ||||||
|  | #Octopus Climbing Device | ||||||
|  |         "OCD Blueprint - Middle Pipe": GrinchLocationData("Who Dump", "Octopus Climbing Device 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)]), | ||||||
|  |         "OCD Blueprint - Mayor's House Rat Vent": GrinchLocationData("Who Dump", "Octopus Climbing Device 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)]), | ||||||
|  |         "OCD Blueprint - Near Power Plant Wall on right side": GrinchLocationData("Who Dump", "Octopus Climbing Device 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)]), | ||||||
|  |         "OCD Blueprint - Guardian's House - Left Side": GrinchLocationData("Minefield", "Octopus Climbing Device 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)]), | ||||||
|  |         "OCD Blueprint - Inside Guardian's House": GrinchLocationData("Minefield", "Octopus Climbing Device Blueprints", 1009, [GrinchRamData(0x01026E, binary_bit_pos=3)]), | ||||||
|  | #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)]), | ||||||
|  |         "MM Blueprint - South Shore - Tent near Porcupine": GrinchLocationData("Who Lake", "Marine Mobile 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)]), | ||||||
|  |         "MM Blueprint - South Shore - Near Hill Bridge": GrinchLocationData("Who Lake", "Marine Mobile 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)]), | ||||||
|  |         "MM Blueprint - South Shore - Grass Platform": GrinchLocationData("Who Lake", "Marine Mobile 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)]), | ||||||
|  |         "MM Blueprint - South Shore - Behind Summer Beast": GrinchLocationData("Who Lake", "Marine Mobile 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)]), | ||||||
|  |         "MM Blueprint - North Shore - Behind Skunk Hut": GrinchLocationData("North Shore", "Marine Mobile 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)]), | ||||||
|  |         "MM Blueprint - North Shore - Fenced in Area": GrinchLocationData("North Shore", "Marine Mobile 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)]), | ||||||
|  |         "MM Blueprint - North Shore - Boulder Box behind Skunk Hut": GrinchLocationData("North Shore", "Marine Mobile 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)]), | ||||||
|  |         "MM Blueprint - North Shore - Crow Platform near Drill House": GrinchLocationData("North Shore", "Marine Mobile Blueprints", 1115, [GrinchRamData(0x010293, binary_bit_pos=1)]), | ||||||
|  |         #Grinch Copter Blueprints | ||||||
|  |         "GC Blueprint - Whoville City Hall - Safe Room": GrinchLocationData("City Hall", "Grinch Copter 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)]), | ||||||
|  |         "GC Blueprint - Whoville Clock Tower - Before Bells": GrinchLocationData("Countdown to X-Mas Clock Tower", "Grinch Copter 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)]), | ||||||
|  |         "GC Blueprint - Who Forest Ski Resort - Inside Dog's Fence": GrinchLocationData("Ski Resort", "Grinch Copter 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)]), | ||||||
|  |         "GC Blueprint - Who Forest Civic Center - Climb across Bat Cave wall": GrinchLocationData("Civic Center", "Grinch Copter 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)]), | ||||||
|  |         "GC Blueprint - Who Dump Power Plant - Max Cave": GrinchLocationData("Power Plant", "Grinch Copter 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)]), | ||||||
|  |         "GC Blueprint - Who Dump Generator Building - Before Mission": GrinchLocationData("Generator Building", "Grinch Copter 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)]), | ||||||
|  |         "GC Blueprint - Who Lake South Shore - Submarine World - Above Surface": GrinchLocationData("Submarine World", "Grinch Copter 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)]), | ||||||
|  |         "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)]), | ||||||
|  |         "GC Blueprint - Who Lake North Shore - Mayor's Villa - Cave": GrinchLocationData("Mayor's Villa", "Grinch Copter Blueprints", 1215, [GrinchRamData(0x010275, binary_bit_pos=6)]), | ||||||
|  | #Sleigh Room Locations | ||||||
|  |         "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)]), | ||||||
|  | #Heart of Stones | ||||||
|  |         "Heart of Stone - Whoville's Post Office": GrinchLocationData("Post Office", "Heart of Stones", 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)]), | ||||||
|  |         "Heart of Stone - Who Dump's Minefield": GrinchLocationData("Minefield", "Heart of Stones", 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)]), | ||||||
|  | #Supadow Minigames | ||||||
|  |         # "Spin N' Win - Easy": GrinchLocationData("Spin N' Win Supadow", "Supadow Minigames", 1500, [GrinchRamData()]), | ||||||
|  |         # "Spin N' Win - Hard": GrinchLocationData("Spin N' Win Supadow", "Supadow Minigames", 1501, [GrinchRamData()]), | ||||||
|  |         # "Spin N' Win - Real Tough": GrinchLocationData("Spin N' Win Supadow", "Supadow Minigames", 1502, [GrinchRamData()]), | ||||||
|  |         # "Dankamania - Easy - 15 Points": GrinchLocationData("Dankamania Supadow", "Supadow Minigames", 1503, [GrinchRamData()]), | ||||||
|  |         # "Dankamania - Hard - 15 Points": GrinchLocationData("Dankamania Supadow", "Supadow Minigames", 1504, [GrinchRamData()]), | ||||||
|  |         # "Dankamania - Real Tough - 15 Points": GrinchLocationData("Dankamania Supadow", "Supadow Minigames", 1505, [GrinchRamData()]), | ||||||
|  |         # "The Copter Race Contest - Easy": GrinchLocationData("The Copter Race Contest Supadow", "Supadow Minigames", 1506, [GrinchRamData()]), | ||||||
|  |         # "The Copter Race Contest - Hard": GrinchLocationData("The Copter Race Contest Supadow", "Supadow Minigames", 1507, [GrinchRamData()]), | ||||||
|  |         # "The Copter Race Contest - Real Tough": GrinchLocationData("The Copter Race Contest Supadow", "Supadow Minigames", 1508, [GrinchRamData()]), | ||||||
|  |         # "Bike Race - 1st Place":  GrinchLocationData("Bike Race", "Supadow Minigames", 1509, [GrinchRamData()]), | ||||||
|  |         # "Bike Race - Top 2": GrinchLocationData("Bike Race", "Supadow Minigames", 1510, [GrinchRamData()]), | ||||||
|  |         # "Bike Race - Top 3": GrinchLocationData("Bike Race", "Supadow Minigames", 1511, [GrinchRamData()]), | ||||||
|  | # Sleigh Part Locations | ||||||
|  |         "Exhaust Pipes in Whoville": GrinchLocationData("Sleigh Room", "Sleigh Ride", 1600, [GrinchRamData(0x0101FB, binary_bit_pos=2)]), | ||||||
|  |         "Skis in Who Forest": GrinchLocationData("Sleigh Room", "Sleigh Ride", 1601, [GrinchRamData(0x0101FB, binary_bit_pos=3)]), | ||||||
|  |         "Tires in Who Dump": GrinchLocationData("Sleigh Room", "Sleigh Ride", 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)]), | ||||||
|  |         "GPS in Who Lake": GrinchLocationData("Sleigh Room", "Sleigh Ride", 1604, [GrinchRamData(0x0101FB, binary_bit_pos=5)]), | ||||||
|  | } | ||||||
|  |  | ||||||
|  | def grinch_locations_to_id() -> dict[str,int]: | ||||||
|  |     location_mappings: dict[str, int] = {} | ||||||
|  |     for LocationName, LocationData in grinch_locations.items(): | ||||||
|  |         location_mappings.update({LocationName: GrinchLocation.get_apid(LocationData.id)}) | ||||||
|  |     return location_mappings | ||||||
							
								
								
									
										90
									
								
								worlds/grinch/Options.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								worlds/grinch/Options.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,90 @@ | |||||||
|  | from dataclasses import dataclass | ||||||
|  |  | ||||||
|  | from Options import FreeText, NumericOption, Toggle, DefaultOnToggle, Choice, TextChoice, Range, NamedRange, OptionList, \ | ||||||
|  |     PerGameCommonOptions | ||||||
|  |  | ||||||
|  | class StartingArea(Choice): | ||||||
|  |     """ | ||||||
|  |     Here, you can select which area you'll start the game with. [NOT IMPLEMENTED] | ||||||
|  |     Whichever one you pick is the region you'll have access to at the start of the Multiworld. | ||||||
|  |     """ | ||||||
|  |     option_whoville = 0 | ||||||
|  |     option_who_forest = 1 | ||||||
|  |     option_who_dump = 2 | ||||||
|  |     option_who_lake = 3 | ||||||
|  |     default = 0 | ||||||
|  |     display_name = "Starting Area" | ||||||
|  |  | ||||||
|  | class ProgressiveVacuum(Toggle):#DefaultOnToggle | ||||||
|  |     """ | ||||||
|  |     Determines whether you get access to main areas progressively [NOT IMPLEMENTED] | ||||||
|  |  | ||||||
|  |     Enabled: Whoville > Who Forest > Who Dump > Who Lake | ||||||
|  |     """ | ||||||
|  |     display_name = "Progressive Vacuum Access" | ||||||
|  |  | ||||||
|  | class Missionsanity(Choice): | ||||||
|  |     """ | ||||||
|  |     How mission checks are randomized in the pool [NOT IMPLEMENTED] | ||||||
|  |  | ||||||
|  |     None: Does not add mission checks | ||||||
|  |     Completion: Only completing the mission gives you a check | ||||||
|  |     Individual: Individual tasks for one mission, such as individual snowmen squashed, are checks. | ||||||
|  |     Both: Both individual tasks and mission completion are randomized. | ||||||
|  |     """ | ||||||
|  |     display_name = "Mission Locations" | ||||||
|  |     option_none = 0 | ||||||
|  |     option_completion = 1 | ||||||
|  |     option_individual = 2 | ||||||
|  |     option_both = 3 | ||||||
|  |     default = 1 | ||||||
|  |  | ||||||
|  | class AnnoyingLocations(DefaultOnToggle): | ||||||
|  |     """Makes certain long, annoying, and tedious checks to be excluded [NOT IMPLEMENTED]""" | ||||||
|  |     display_name = "Annoying Locations" | ||||||
|  |  | ||||||
|  | class ProgressiveGadget(Toggle):#DefaultOnToggle | ||||||
|  |     """ | ||||||
|  |     Determines whether you get access to a gadget as individual blueprint count [NOT IMPLEMENTED] | ||||||
|  |     """ | ||||||
|  |     display_name = "Progressive Gadgets" | ||||||
|  |  | ||||||
|  | class Supadow(Toggle): | ||||||
|  |     """Enables completing minigames through the Supadows in Mount Crumpit as checks. (9 locations) [NOT IMPLEMENTED]""" | ||||||
|  |     display_name = "Supadow Minigame Locations" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Gifts(Toggle): | ||||||
|  |     """Missions that require you to squash every present in a level. (4 locations) [NOT IMPLEMENTED]""" | ||||||
|  |     display_name = "Gift Collection Locations" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Movesanity(Toggle): | ||||||
|  |     """Randomizes Grinch's moveset along with randomizing max into the pool. [NOT IMPLEMENTED]""" | ||||||
|  |     display_name = "Movesanity" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class UnlimitedEggs(Toggle): | ||||||
|  |     """Determine whether or not you run out of rotten eggs when you utilize your gadgets.""" | ||||||
|  |     display_name = "Unlimited Rotten Eggs" | ||||||
|  |  | ||||||
|  | class RingLinkOption(Toggle): | ||||||
|  |     """Whenever this is toggled, your ammo is linked with other ringlink-compatible games that also have this enabled. [NOT IMPLEMENTED]""" | ||||||
|  |  | ||||||
|  | class TrapLinkOption(Toggle): | ||||||
|  |     """If a trap is sent from Grinch, traps that are compatible with other games are triggered aswell. [NOT IMPLEMENTED]""" | ||||||
|  |  | ||||||
|  | @dataclass | ||||||
|  | class GrinchOptions(PerGameCommonOptions):#DeathLinkMixin | ||||||
|  |     starting_area: StartingArea | ||||||
|  |     progressive_vacuum: ProgressiveVacuum | ||||||
|  |     missionsanity: Missionsanity | ||||||
|  |     annoying_locations: AnnoyingLocations | ||||||
|  |     minigamesanity: Supadow | ||||||
|  |     progressive_gadget: ProgressiveGadget | ||||||
|  |     supadow_minigames: Supadow | ||||||
|  |     giftsanity: Gifts | ||||||
|  |     movesanity: Movesanity | ||||||
|  |     unlimited_eggs: UnlimitedEggs | ||||||
|  |     ring_link: RingLinkOption | ||||||
|  |     trap_link: TrapLinkOption | ||||||
							
								
								
									
										12
									
								
								worlds/grinch/RamHandler.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								worlds/grinch/RamHandler.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | from typing import NamedTuple, Optional | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GrinchRamData(NamedTuple): | ||||||
|  |     ram_address: int | ||||||
|  |     value: Optional[int] = None #none is empty/null | ||||||
|  |     # Either set or add either hex or unsigned values through Client.py | ||||||
|  |     # Hex uses 0x00, unsigned are base whole numbers | ||||||
|  |     binary_bit_pos: Optional[int] = None | ||||||
|  |     bit_size: int = 1 | ||||||
|  |     update_existing_value: bool = False | ||||||
|  |     max_count: int = 0 | ||||||
							
								
								
									
										101
									
								
								worlds/grinch/Regions.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								worlds/grinch/Regions.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,101 @@ | |||||||
|  | from typing import TYPE_CHECKING | ||||||
|  |  | ||||||
|  | from BaseClasses import Region | ||||||
|  | from .Rules import access_rules_dict, interpret_rule | ||||||
|  |  | ||||||
|  | from ..generic.Rules import add_rule | ||||||
|  |  | ||||||
|  | if TYPE_CHECKING: | ||||||
|  |  | ||||||
|  |     from . import GrinchWorld | ||||||
|  |  | ||||||
|  | mainareas_list = [ | ||||||
|  |     "Whoville", | ||||||
|  |     "Who Forest", | ||||||
|  |     "Who Dump", | ||||||
|  |     "Who Lake" | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | subareas_list = [ | ||||||
|  |     "Post Office", | ||||||
|  |     "City Hall", | ||||||
|  |     "Countdown to X-Mas Tower", | ||||||
|  |     "Ski Resort", | ||||||
|  |     "Civic Center", | ||||||
|  |     "Minefield", | ||||||
|  |     "Power Plant", | ||||||
|  |     "Generator Building", | ||||||
|  |     "Submarine World", | ||||||
|  |     "Scout's Hut", | ||||||
|  |     "North Shore", | ||||||
|  |     "Mayor's Villa", | ||||||
|  |     "Sleigh Room" | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | supadow_list = [ | ||||||
|  |     "Spin N' Win Supadow", | ||||||
|  |     "Dankamania Supadow", | ||||||
|  |     "The Copter Race Contest Supadow", | ||||||
|  |     "Bike Race" | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | def create_regions(world: "GrinchWorld"): | ||||||
|  |     for mainarea in mainareas_list: | ||||||
|  |         #Each area in mainarea, create a region for the given player | ||||||
|  |         world.multiworld.regions.append(Region(mainarea, world.player, world.multiworld)) | ||||||
|  |     for subarea in subareas_list: | ||||||
|  |         #Each area in subarea, create a region for the given player | ||||||
|  |         world.multiworld.regions.append(Region(subarea, world.player, world.multiworld)) | ||||||
|  |     for supadow in supadow_list: | ||||||
|  |         #Each area in supadow, create a region for the given player | ||||||
|  |         world.multiworld.regions.append(Region(supadow, world.player, world.multiworld)) | ||||||
|  |  | ||||||
|  | # TODO Optimize this function | ||||||
|  | def grinchconnect(world: "GrinchWorld", current_region_name: str, connected_region_name: str): | ||||||
|  |     current_region = world.get_region(current_region_name) | ||||||
|  |     connected_region = world.get_region(connected_region_name) | ||||||
|  |     required_items: list[list[str]] = access_rules_dict[connected_region.name] | ||||||
|  |     rule_list = interpret_rule(required_items, world.player) | ||||||
|  |     # Goes from current to connected | ||||||
|  |     current_region.connect(connected_region) | ||||||
|  |     # Goes from connected to current | ||||||
|  |     connected_region.connect(current_region) | ||||||
|  |     for access_rule in rule_list: | ||||||
|  |         for region_entrance in current_region.entrances: | ||||||
|  |             if region_entrance.connected_region.name == current_region_name and \ | ||||||
|  |                 region_entrance.parent_region.name == connected_region_name: | ||||||
|  |                 if rule_list.index(access_rule) == 0: | ||||||
|  |                     add_rule(region_entrance, access_rule) | ||||||
|  |                 else: | ||||||
|  |                     add_rule(region_entrance, access_rule, combine="or") | ||||||
|  |         for region_entrance in connected_region.entrances: | ||||||
|  |             if region_entrance.connected_region.name == connected_region_name and \ | ||||||
|  |                     region_entrance.parent_region.name == current_region_name: | ||||||
|  |                 if rule_list.index(access_rule) == 0: | ||||||
|  |                     add_rule(region_entrance, access_rule) | ||||||
|  |                 else: | ||||||
|  |                     add_rule(region_entrance, access_rule, combine="or") | ||||||
|  |  | ||||||
|  | #What regions are connected to each other | ||||||
|  | def connect_regions(world: "GrinchWorld"): | ||||||
|  |     grinchconnect(world, "Mount Crumpit", "Whoville") | ||||||
|  |     grinchconnect(world, "Mount Crumpit", "Who Forest") | ||||||
|  |     grinchconnect(world, "Mount Crumpit", "Who Dump") | ||||||
|  |     grinchconnect(world, "Mount Crumpit", "Who Lake") | ||||||
|  |     grinchconnect(world, "Mount Crumpit", "Sleigh Room") | ||||||
|  |     grinchconnect(world, "Mount Crumpit", "Spin N' Win Supadow") | ||||||
|  |     grinchconnect(world, "Mount Crumpit", "Dankamania Supadow") | ||||||
|  |     grinchconnect(world, "Mount Crumpit", "The Copter Race Contest Supadow") | ||||||
|  |     grinchconnect(world, "Whoville", "Post Office") | ||||||
|  |     grinchconnect(world, "Whoville", "City Hall") | ||||||
|  |     grinchconnect(world, "Whoville", "Countdown to X-Mas Clock Tower") | ||||||
|  |     grinchconnect(world, "Who Forest", "Ski Resort") | ||||||
|  |     grinchconnect(world, "Who Forest", "Civic Center") | ||||||
|  |     grinchconnect(world, "Who Dump", "Minefield") | ||||||
|  |     grinchconnect(world, "Who Dump", "Power Plant") | ||||||
|  |     grinchconnect(world, "Power Plant", "Generator Building") | ||||||
|  |     grinchconnect(world, "Who Lake", "Submarine World") | ||||||
|  |     grinchconnect(world, "Who Lake", "Scout's Hut") | ||||||
|  |     grinchconnect(world, "Who Lake", "North Shore") | ||||||
|  |     grinchconnect(world, "North Shore", "Mayor's Villa") | ||||||
|  |     grinchconnect(world, "Sleigh Room", "Bike Race") | ||||||
							
								
								
									
										578
									
								
								worlds/grinch/Rules.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										578
									
								
								worlds/grinch/Rules.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,578 @@ | |||||||
|  | from typing import Callable | ||||||
|  |  | ||||||
|  | import Utils | ||||||
|  | from BaseClasses import CollectionState | ||||||
|  | from worlds.AutoWorld import World | ||||||
|  | from worlds.generic.Rules import add_rule | ||||||
|  |  | ||||||
|  | #Adds all rules from access_rules_dict to locations | ||||||
|  | def set_location_rules(world: World): | ||||||
|  |     all_locations = world.get_locations() | ||||||
|  |     for location in all_locations: | ||||||
|  |         loc_rules = rules_dict[location.name] | ||||||
|  |         rule_list = interpret_rule(loc_rules, world.player) | ||||||
|  |         for access_rule in rule_list: | ||||||
|  |             if rule_list.index(access_rule) == 0: | ||||||
|  |                 add_rule(location, access_rule) | ||||||
|  |             else: | ||||||
|  |                 add_rule(location, access_rule, "or") | ||||||
|  |  | ||||||
|  | def interpret_rule(rule_set: list[list[str]], player: int): | ||||||
|  |     # If a region/location does not have any items required, make the section(s) return no logic. | ||||||
|  |     if len(rule_set) < 1: | ||||||
|  |         return True | ||||||
|  |  | ||||||
|  |     # Otherwise, if a region/location DOES have items required, make the section(s) return list of logic. | ||||||
|  |  | ||||||
|  |     access_list: list[Callable[[CollectionState], bool]] = [] | ||||||
|  |     for item_set in rule_set: | ||||||
|  |         access_list.append(lambda state: state.has_all(item_set, player)) | ||||||
|  |     return access_list | ||||||
|  |  | ||||||
|  |     #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]]] = { | ||||||
|  |     "Enter Whoville": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter the Post Office": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter the Town Hall": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter the Countdown-To-Xmas Clock Tower": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter Who Forest": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter the Ski Resort": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter the Civic Center": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter Who Dump": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter the Minefield": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter the Power Plant": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter the Generator Building": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter Who Lake": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter the Submarine World": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter the Scout's Hut": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter the North Shore": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Enter the Mayor's Villa": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Shuffling The Mail": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Smashing Snowmen": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Painting The Mayor's Posters": [ | ||||||
|  |         ["Painting Bucket"] | ||||||
|  |     ], | ||||||
|  |     "Launching Eggs Into Houses": [ | ||||||
|  |         ["Rotten Egg Launcher"] | ||||||
|  |     ], | ||||||
|  |     "Modifying The Mayor's Statue": [ | ||||||
|  |         ["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": [ | ||||||
|  |         ["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": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Binoculars Blueprint - City Hall Library - Front Side": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Binoculars Blueprint - City Hall Library - Right Side": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "REL Blueprint - Outside City Hall": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "REL Blueprint - Outside Clock Tower": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "REL Blueprint - Post Office - Inside Silver Room": [ | ||||||
|  |         ["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": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "RS Blueprint - Near Tree House on Ground": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "RS Blueprint - Near Cable Car House": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "RS Blueprint - Near Who Snowball in Cave": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "RS Blueprint - Branch Platform Closest to Glue Cannon": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "RS Blueprint - Branch Platform Near Beast": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "RS Blueprint - Branch Platform Ledge Grab House": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "RS Blueprint - On Tree House": [ | ||||||
|  |         ["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": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "MM Blueprint - South Shore - Near Outhouse": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "MM Blueprint - South Shore - Near Hill Bridge": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "MM Blueprint - South Shore - Scout's Hut Roof": [ | ||||||
|  |         ["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": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "MM Blueprint - North Shore - Behind Skunk Hut": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "MM Blueprint - North Shore - Inside Skunk Hut": [ | ||||||
|  |         [] | ||||||
|  |         # ["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": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "MM Blueprint - North Shore - Inside Drill House": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "MM Blueprint - North Shore - Crow Platform near Drill House": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Whoville City Hall - Safe Room": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Whoville City Hall - Statue Room": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Whoville Clock Tower - Before Bells": [ | ||||||
|  |         ["Rocket Spring"] | ||||||
|  |     #   ["Max", "Rocket Spring"] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Whoville Clock Tower - After Bells": [ | ||||||
|  |         ["Rocket Spring"] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Who Forest Ski Resort - Inside Dog's Fence": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Who Forest Ski Resort - Max Cave": [ | ||||||
|  |         [] | ||||||
|  |         # ["Max"] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Who Forest Civic Center - Climb across Bat Cave wall": [ | ||||||
|  |         ["Grinch Copter"], | ||||||
|  |         ["Octopus Climbing Device", "Rocket Spring"] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Who Forest Civic Center - Shoot Icicle in Bat Entrance": [ | ||||||
|  |         ["Rotten Egg Launcher", "Grinch Copter"], | ||||||
|  |         ["Rotten Egg Launcher", "Octopus Climbing Device", "Rocket Spring"], | ||||||
|  |         ["Slime Shooter", "Grinch Copter"], | ||||||
|  |         ["Slime Shooter", "Octopus Climbing Device", "Rocket Spring"] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Who Dump Power Plant - Max Cave": [ | ||||||
|  |         [] | ||||||
|  |         # ["Max"] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Who Dump Power Plant - After First Gate": [ | ||||||
|  |         ["Rotten Egg Launcher", "Rocket Spring"], | ||||||
|  |         ["Grinch Copter"] | ||||||
|  |     #   ["Max", "Rotten Egg Launcher", "Rocket Spring"] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Who Dump Generator Building - Before Mission": [ | ||||||
|  |         ["Rotten Egg Launcher", "Grinch Copter"], | ||||||
|  |         ["Rotten Egg Launcher", "Octopus Climbing Device", "Slime Shooter", "Rocket Spring"] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Who Dump Generator Building - After Mission": [ | ||||||
|  |         ["Rotten Egg Launcher", "Grinch Copter"], | ||||||
|  |         ["Rotten Egg Launcher", "Octopus Climbing Device", "Slime Shooter", "Rocket Spring"] | ||||||
|  |     ], | ||||||
|  |     "GC Blueprint - Who Lake South Shore - Submarine World - Above Surface": [ | ||||||
|  |         ["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"] | ||||||
|  |         ["Rotten Egg Launcher", "Who Forest Vacuum Access", "Who Dump Vacuum Access", "Who Lake Vacuum Access", "Rocket Spring", "Marine Mobile"] | ||||||
|  |     ], | ||||||
|  |     "Neutralizing Santa": [ | ||||||
|  |         # ["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"] | ||||||
|  |     ], | ||||||
|  |     "Heart of Stone - Whoville's Post Office": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Heart of Stone - Who Forest's Ski Resort": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Heart of Stone - Who Dump's Minefield": [ | ||||||
|  |         ["Grinch Copter"], | ||||||
|  |         ["Rotten Egg Launcher", "Slime Shooter", "Rocket Spring"] | ||||||
|  |     ], | ||||||
|  |     "Heart of Stone - Who Lake's North Shore": [ | ||||||
|  |         [] | ||||||
|  |         # ["Max"] | ||||||
|  |     ], | ||||||
|  |     "Spin N' Win - Easy": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Spin N' Win - Hard": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Spin N' Win - Real Tough": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Dankamania - Easy - 15 Points": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Dankamania - Hard - 15 Points": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Dankamania - Real Tough - 15 Points": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "The Copter Race Contest - Easy": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "The Copter Race Contest - Hard": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "The Copter Race Contest - Real Tough": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Bike Race - 1st Place": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Bike Race - Top 2": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Bike Race - Top 3": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Exhaust Pipes in Whoville": [ | ||||||
|  |         ["Rotten Egg Launcher"] | ||||||
|  |     ], | ||||||
|  |     "Skis in Who Forest": [ | ||||||
|  |         ["Who Forest Vacuum Access"] | ||||||
|  |     ], | ||||||
|  |     "Tires in Who Dump": [ | ||||||
|  |         ["Who Dump Vacuum Access", "Rocket Spring", "Rotten Egg Launcher"] | ||||||
|  |     ], | ||||||
|  |     "Twin-End Tuba in Submarine World": [ | ||||||
|  |         ["Who Lake Vacuum Access", "Marine Mobile"] | ||||||
|  |     ], | ||||||
|  |     "GPS in Who Lake": [ | ||||||
|  |         ["Who Lake Vacuum Access", "Rotten Egg Launcher"] | ||||||
|  |     ], | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | access_rules_dict: dict[str,list[list[str]]] = { | ||||||
|  |     "Whoville": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Post Office": [ | ||||||
|  |         ["Who Cloak"] | ||||||
|  |     ], | ||||||
|  |     "City Hall": [ | ||||||
|  |         ["Rotten Egg Launcher"] | ||||||
|  |     ], | ||||||
|  |     "Countdown to X-Mas Clock Tower": [ | ||||||
|  |         [] | ||||||
|  |     ], | ||||||
|  |     "Who Forest": [ | ||||||
|  |         ["Who Forest Vacuum Access"], | ||||||
|  |         # ["Progressive Vacuum Access": 1] | ||||||
|  |     ], | ||||||
|  |     "Ski Resort": [ | ||||||
|  |         ["Cable Car Access Card"] | ||||||
|  |     ], | ||||||
|  |     "Civic Center": [ | ||||||
|  |         ["Grinch Copter"], | ||||||
|  |         ["Octopus Climbing Device"] | ||||||
|  |     ], | ||||||
|  |     "Who Dump": [ | ||||||
|  |         ["Who Dump Vacuum Access"], | ||||||
|  |         # ["Progressive Vacuum Access": 2] | ||||||
|  |     ], | ||||||
|  |     "Minefield": [ | ||||||
|  |         ["Rotten Egg Launcher", "Slime Shooter", "Rocket Spring"], | ||||||
|  |         ["Rotten Egg Launcher", "Grinch Copter"] | ||||||
|  |     ], | ||||||
|  |     "Power Plant": [ | ||||||
|  |         ["Rotten Egg Launcher", "Grinch Copter"], | ||||||
|  |         ["Slime Shooter", "Grinch Copter"], | ||||||
|  |         ["Rotten Egg Launcher", "Octopus Climbing Device", "Slime Shooter", "Rocket Spring"] | ||||||
|  |     ], | ||||||
|  |     "Generator Building": [ | ||||||
|  |         ["Rotten Egg Launcher", "Grinch Copter"], | ||||||
|  |         ["Rotten Egg Launcher", "Octopus Climbing Device", "Slime Shooter", "Rocket Spring"] | ||||||
|  |     ], | ||||||
|  |     "Who Lake": [ | ||||||
|  |         ["Who Lake Vacuum Access"], | ||||||
|  |         # ["Progressive Vacuum Access": 3] | ||||||
|  |     ], | ||||||
|  |     "Scout's Hut": [ | ||||||
|  |         ["Grinch Copter"], | ||||||
|  |         ["Rocket Spring"] | ||||||
|  |     ], | ||||||
|  |     "North Shore": [ | ||||||
|  |         ["Scout Clothes"] | ||||||
|  |     ], | ||||||
|  |     "Mayor's Villa": [ | ||||||
|  |         ["Scout Clothes"] | ||||||
|  |     ], | ||||||
|  |     "Submarine World": [ | ||||||
|  |         ["Marine Mobile"] | ||||||
|  |     ], | ||||||
|  |     "Sleigh Room": [ | ||||||
|  |         ["Sleigh Room Key"] | ||||||
|  |     ], | ||||||
|  |     "Spin N' Win Supadow": [ | ||||||
|  |         ["Spin N' Win Door Unlock"], | ||||||
|  |         # ["Progressive Supadow Door Unlock"] | ||||||
|  |     ], | ||||||
|  |     "Dankamania Supadow": [ | ||||||
|  |         ["Dankamania Door Unlock"], | ||||||
|  |         # ["Progressive Supadow Door Unlock: 2"] | ||||||
|  |     ], | ||||||
|  |     "The Copter Race Contest Supadow": [ | ||||||
|  |         ["The Copter Race Contest Door Unlock"], | ||||||
|  |         # ["Progressive Supadow Door Unlock: 3"] | ||||||
|  |     ], | ||||||
|  |     "Bike Race": [ | ||||||
|  |         ["Bike Race Access"], | ||||||
|  |         # ["Progressive Supadow Door Unlock: 4"] | ||||||
|  |     ] | ||||||
|  | } | ||||||
							
								
								
									
										80
									
								
								worlds/grinch/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								worlds/grinch/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,80 @@ | |||||||
|  | from BaseClasses import Region, Item, ItemClassification | ||||||
|  | from .Locations import grinch_locations_to_id, grinch_locations, GrinchLocation | ||||||
|  | from .Items import grinch_items_to_id, GrinchItem, ALL_ITEMS_TABLE, MISC_ITEMS_TABLE | ||||||
|  | from .Regions import connect_regions | ||||||
|  | from .Rules import set_location_rules | ||||||
|  |  | ||||||
|  | from .Client import * | ||||||
|  | from typing import ClassVar | ||||||
|  |  | ||||||
|  | from worlds.AutoWorld import World | ||||||
|  |  | ||||||
|  | from . import Options | ||||||
|  | from .Rules import access_rules_dict | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GrinchWorld(World): | ||||||
|  |     game: ClassVar[str] = "The Grinch" | ||||||
|  |     options_dataclass = Options.GrinchOptions | ||||||
|  |     options = Options.GrinchOptions | ||||||
|  |     topology_present = True #not an open world game, very linear | ||||||
|  |     item_name_to_id: ClassVar[dict[str,int]] = grinch_items_to_id() | ||||||
|  |     location_name_to_id: ClassVar[dict[str,int]] = grinch_locations_to_id() | ||||||
|  |     required_client_version = (0, 6, 3) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def __init__(self, *args, **kwargs): #Pulls __init__ function and takes control from there in BaseClasses.py | ||||||
|  |         self.origin_region_name: str = "Mount Crumpit" | ||||||
|  |         super(GrinchWorld, self).__init__(*args, **kwargs) | ||||||
|  |  | ||||||
|  |     def generate_early(self) -> None: #Special conditions changed before generation occurs | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     def create_regions(self): #Generates all regions for the multiworld | ||||||
|  |         for region_name in access_rules_dict.keys(): | ||||||
|  |             self.multiworld.regions.append(Region(region_name, self.player, self.multiworld)) | ||||||
|  |         self.multiworld.regions.append(Region("Mount Crumpit", self.player, self.multiworld)) | ||||||
|  |         for location, data in grinch_locations.items(): | ||||||
|  |             region = self.get_region(data.region) | ||||||
|  |             entry = GrinchLocation(self.player, location, region, data) | ||||||
|  |             if location == "Neutralizing Santa": | ||||||
|  |                 entry.place_locked_item(Item("Goal", ItemClassification.progression, None, self.player)) | ||||||
|  |             region.locations.append(entry) | ||||||
|  |         connect_regions(self) | ||||||
|  |  | ||||||
|  |     def create_item(self, item: str) -> GrinchItem: #Creates specific items on demand | ||||||
|  |         if item in ALL_ITEMS_TABLE.keys(): | ||||||
|  |             return GrinchItem(item, self.player, ALL_ITEMS_TABLE[item]) | ||||||
|  |         raise Exception(f"Invalid item name: {item}") | ||||||
|  |  | ||||||
|  |     def create_items(self): #Generates all items for the multiworld | ||||||
|  |         self_itempool: list[GrinchItem] = [] | ||||||
|  |         for item, data in ALL_ITEMS_TABLE.items(): | ||||||
|  |             self_itempool.append(self.create_item(item)) | ||||||
|  |             if item == "Heart of Stone": | ||||||
|  |                 for _ in range(3): | ||||||
|  |                     self_itempool.append(self.create_item(item)) | ||||||
|  |  | ||||||
|  |         #Get number of current unfilled locations | ||||||
|  |         unfilled_locations: int = len(self.multiworld.get_unfilled_locations(self.player)) - len(ALL_ITEMS_TABLE.keys()) - 3 | ||||||
|  |  | ||||||
|  |         for _ in range(unfilled_locations): | ||||||
|  |             self_itempool.append(self.create_item((self.get_other_filler_item(list(MISC_ITEMS_TABLE.keys()))))) | ||||||
|  |         self.multiworld.itempool += self_itempool | ||||||
|  |  | ||||||
|  |     def set_rules(self): | ||||||
|  |         self.multiworld.completion_condition[self.player] = lambda state: state.has("Goal", self.player) | ||||||
|  |         set_location_rules(self) | ||||||
|  |  | ||||||
|  |     def get_other_filler_item(self, other_filler: list[str]) -> str: | ||||||
|  |         return self.random.choices(other_filler)[0] | ||||||
|  |  | ||||||
|  |     def fill_slot_data(self): | ||||||
|  |         return { | ||||||
|  |             "give_unlimited_eggs": self.options.unlimited_eggs.value, | ||||||
|  |  | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     def generate_output(self, output_directory: str) -> None: | ||||||
|  |         # print("") | ||||||
|  |         pass | ||||||
							
								
								
									
										5
									
								
								worlds/grinch/docs/Credits.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								worlds/grinch/docs/Credits.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | - 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. | ||||||
|  | - Shoutouts to SomeJakeGuy for basically teaching me how to code in general. | ||||||
|  | - 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. | ||||||
							
								
								
									
										26
									
								
								worlds/grinch/docs/en_The Grinch.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								worlds/grinch/docs/en_The Grinch.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | # The Grinch | ||||||
|  |  | ||||||
|  | ## Where is the options page? | ||||||
|  |  | ||||||
|  | The [player options page for this game](../player-options) contains all the | ||||||
|  | options you need to configure and export a config file. | ||||||
|  |  | ||||||
|  | ## What items and locations get shuffled? | ||||||
|  |  | ||||||
|  | Items such as gadgets, sleigh parts, heart of stones, vacuum accesses, and useful items (or mission specific items) are  | ||||||
|  | shuffled. Locations that players can do to send items out include visiting an area for the first time, collecting a blueprint,  | ||||||
|  | and completing a mission. | ||||||
|  |  | ||||||
|  | ## What is the goal of The Grinch when randomized? | ||||||
|  |  | ||||||
|  | The player must obtain the Skis, Twin-End Tuba, Exhaust Pipes, and the Tires to be able to build the sleigh and to finally | ||||||
|  | stop Christmas from coming once and for all by neutralizing Santa Claus! | ||||||
|  |  | ||||||
|  | ## Which items can be in another player's world? | ||||||
|  |  | ||||||
|  | - All gadgets such as the binoculars, Rotten Egg Launcher, Rocket Spring, Slime Shooter, Octopus Climbing Device, Marine  | ||||||
|  | Mobile, and the Grinch Copter. | ||||||
|  | - 4 Heart of Stones that increases the player's max health by one. | ||||||
|  | - Useful items, or mission specific items, that allow the player to access certain subareas or complete missions that  | ||||||
|  | require that specific item. | ||||||
|  | - Vacuum accesses that give the player access to Who Forest, Who Dump, and Who Lake respectively. | ||||||
							
								
								
									
										54
									
								
								worlds/grinch/docs/setup_en.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								worlds/grinch/docs/setup_en.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | |||||||
|  | # The Grinch - Setup Guide | ||||||
|  |  | ||||||
|  | ## Required Software | ||||||
|  | - [Archipelago](https://github.com/ArchipelagoMW/Archipelago/releases). Please use version 0.6.2 or later for integrated | ||||||
|  | BizHawk support. | ||||||
|  | - 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. | ||||||
|  | - [BizHawk](https://tasvideos.org/BizHawk/ReleaseHistory) Version 2.9.1 is supported, but I can't promise if any version is stable or not. | ||||||
|  | - 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. | ||||||
|  |  | ||||||
|  | ## Configuring your Config (.yaml) file | ||||||
|  |  | ||||||
|  | ### What is a config file and why do I need one? | ||||||
|  |  | ||||||
|  | See the guide on setting up a basic YAML at the Archipelago setup | ||||||
|  | guide: [Basic Multiworld Setup Guide](/tutorial/Archipelago/setup/en) | ||||||
|  |  | ||||||
|  | ### Where do I get a config file? | ||||||
|  |  | ||||||
|  | The Player options page on the website allows you to configure your personal | ||||||
|  | options and export a config file from them: [The Grinch Player Options Page](../player-options) | ||||||
|  |  | ||||||
|  | ### 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 | ||||||
|  |  | ||||||
|  | Once both the client and the emulator are started, they must be connected. **This should happen automatically.** | ||||||
|  | However, if the lua script window doesn't appear, then within the emulator click | ||||||
|  | on the "Tools" menu and select "Lua Console". Click the folder button or press Ctrl+O to open a Lua | ||||||
|  | script. Navigate to your Archipelago install folder and open `data/lua/connector_bizhawk_generic.lua`. | ||||||
|  |  | ||||||
|  | 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 | ||||||
|  | `/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