ref: 1e6f29557348eee31cdcf2bcaadbcd93dfe236bf
dir: /extras/old_parse_scripts.py/
# -*- coding: utf-8 -*- """ This is in a separate file because it screws up highlighting. Doesn't do anything on its own. old_parse is a part of the Script class """ def old_parse(self, *args, **kwargs): """parses a script-engine script; force=True if you want to re-parse and get the debug information""" print "Script.old_parse address="+hex(self.address) #can't handle more than one argument if len(args) > 1: raise Exception, "Script.parse_script doesn't know how to handle positional arguments" #use the first positional argument as the address elif len(args) == 1: self.address = args[0] if type(self.address) == str: self.address = int(self.address, 16) elif type(self.address) != int: raise Exception, "address param is the wrong type" #parse any keyword arguments, first make up the defaults kwargsorig = {"map_group": None, "map_id": None, "force": False, "debug": True, "origin": False} #let the caller override any defaults kwargsorig.update(kwargs) #copy these into kwargs kwargs = kwargsorig #set these defaults map_group = kwargs["map_group"] map_id = kwargs["map_id"] force = kwargs["force"] debug = kwargs["debug"] origin = kwargs["origin"] self.map_group = map_group self.map_id = map_id global rom if rom == None: direct_load_rom() #max number of commands in a 'recursive' script max_cmds = 150 #set the address to be parsed address = self.address original_start_address = address #don't parse these crazy things (battle tower things, some rival things, etc.) if address in stop_points: print "got " + hex(address) + ".. map_group=" + str(map_group) + " map_id=" + str(map_id) return None #don't parse anything that looks crazy if address < 0x4000 and address not in [0x26ef, 0x114, 0x1108]: print "address is less than 0x4000.. address is: " + hex(address) sys.exit() #check if work is being repeated if is_script_already_parsed_at(address) and not force: raise Exception, "this script has already been parsed before, please use that instance" #use the commands from a previously-parsed Script object #self.commands = script_parse_table[address].commands #return True #return a previously-created Script object #return script_parse_table[address] #this next line stops the same script from being re-parsed multiple times #for instance.. maybe there's a script jump, then a jump back #the original script should only be parsed once script_parse_table[original_start_address:original_start_address+1] = "incomplete Script" #set up some variables self.commands = {} commands = self.commands offset = address end = False #main loop.. parse each command byte while not end: #reset variables so we don't contaminate this command info, long_info, size = None, None, 0 #read the current command byte command_byte = ord(rom[offset]) #setup the current command representation command = {"type": command_byte, "start_address": offset} #size is the total size including the command byte #last_byte_address is offset+size-1 start_address = offset if (len(commands.keys()) > max_cmds) and origin != False: print "too many commands in this script? might not be a script (starting at: " +\ hex(original_start_address) + ").. called from a script at: " + hex(origin) sys.exit() #start checking against possible command bytes if command_byte == 0x00: #Pointer code [2b+ret] pksv_name = "2call" info = "pointer code" long_info = """ 2byte pointer points to script; when pointed script ends --> return to old script [code][2 byte pointer] """ size = 3 start_address = offset last_byte_address = offset + size - 1 pointer = calculate_pointer_from_bytes_at(start_address+1) if pointer == None: raise Exception, "pointer is None (shouldn't be None pointer on 0x0 script command" command["pointer"] = pointer if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(pointer, original_start_address, debug=debug) command["script"] = script elif command_byte == 0x01: #Pointer code [3b+ret] pksv_name = "3call" info = "pointer code" long_info = """ 3byte pointer points to script; when pointed script ends --> return to old script [Code][resp. pointer(2byte or 3byte)] """ size = 4 info = "pointer code" pointer = calculate_pointer_from_bytes_at(start_address+1, bank=True) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(pointer, original_start_address, debug) command["pointer"] = pointer command["script"] = script elif command_byte == 0x02: #Pointer code [2b+3b+ret] info = "pointer code" long_info = """ 2byte pointer points to 3byte pointer; when pointed script --> return to old script [Code][resp. pointer(2byte or 3byte)] """ size = 3 pointer = calculate_pointer_from_bytes_at(start_address+1) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(pointer, original_start_address, debug=debug) command["pointer"] = pointer command["script"] = script elif command_byte == 0x03: #Pointer code [2b] #XXX what does "new script is part of main script" mean? info = "pointer code" long_info = """ 2byte pointer points to script; new script is part of main script [Code][resp. pointer(2byte or 3byte)] """ size = 3 pointer = calculate_pointer_from_bytes_at(start_address+1) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(pointer, original_start_address, debug=debug) command["pointer"] = pointer command["script"] = script end = True #according to pksv elif command_byte == 0x04: #Pointer code [3b] info = "pointer code" long_info = """ 3byte pointer points to script; new script is part of main script [Code][resp. pointer(2byte or 3byte)] """ size = 4 pointer = calculate_pointer_from_bytes_at(start_address+1, bank=True) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(pointer, original_start_address, debug=debug) command["pointer"] = pointer command["script"] = script end = True #according to pksv elif command_byte == 0x05: #Pointer code [2b+3b] info = "pointer code" long_info = """ 2byte pointer points to 3byte pointer; new script is part of main script [Code][resp. pointer(2byte or 3byte)] """ size = 3 command["pointer"] = calculate_pointer_from_bytes_at(start_address+1) command["target_pointer"] = calculate_pointer_from_bytes_at(command["pointer"], bank=True) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(command["target_pointer"])+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(command["target_pointer"], original_start_address, debug=debug) command["script"] = script end = True #according to pksv elif command_byte == 0x06: #RAM check [=byte] info = "RAM check [=byte]" long_info = """ When the conditional is true... .. then go to pointed script, else resume interpreting after the pointer """ size = 4 command["byte"] = ord(rom[start_address+1]) pointer = calculate_pointer_from_bytes_at(start_address+2) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(pointer, original_start_address, debug=debug) command["pointer"] = pointer command["script"] = script elif command_byte == 0x07: #RAM check [<>byte] info = "RAM check [<>byte]" long_info = """ When the conditional is true... .. then go to pointed script, else resume interpreting after the pointer """ size = 4 command["byte"] = ord(rom[start_address+1]) pointer = calculate_pointer_from_bytes_at(start_address+2) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(pointer, original_start_address, debug=debug) command["pointer"] = pointer command["script"] = script elif command_byte == 0x08: #RAM check [=0] info = "RAM check [=0]" long_info = """ When the conditional is true... .. then go to pointed script, else resume interpreting after the pointer """ size = 3 pointer = calculate_pointer_from_bytes_at(start_address+1) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(pointer, original_start_address, debug=debug) command["pointer"] = pointer command["script"] = script elif command_byte == 0x09: #RAM check [<>0] info = "RAM check [<>0]" long_info = """ When the conditional is true... .. then go to pointed script, else resume interpreting after the pointer """ size = 3 pointer = calculate_pointer_from_bytes_at(start_address+1) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(pointer, original_start_address, debug=debug) command["pointer"] = pointer command["script"] = script elif command_byte == 0x0A: #RAM check [<byte] info = "RAM check [<byte]" long_info = """ When the conditional is true... .. then go to pointed script, else resume interpreting after the pointer """ size = 4 command["byte"] = ord(rom[start_address+1]) pointer = calculate_pointer_from_bytes_at(start_address+2) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(pointer, original_start_address, debug=debug) command["pointer"] = pointer command["script"] = script elif command_byte == 0x0B: #RAM check [>byte] info = "RAM check [>byte]" long_info = """ When the conditional is true... .. then go to pointed script, else resume interpreting after the pointer """ size = 4 command["byte"] = ord(rom[start_address+1]) pointer = calculate_pointer_from_bytes_at(start_address+2) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(pointer, original_start_address, debug=debug) command["pointer"] = pointer command["script"] = script elif command_byte == 0x0C: #0C codes [xxyy] info = "call predefined script then end" long_info = """ Calls predefined scripts. After this code the script ends. [0C][xxyy] """ size = 3 end = True byte1 = ord(rom[offset+1]) byte2 = ord(rom[offset+2]) number = byte1 + (byte2 << 8) #0000 to 000AD ... XXX how should these be handled? command["predefined_script_number"] = number elif command_byte == 0x0D: #0D codes [xxyy] info = "call some predefined script" long_info = """ Calls predefined scripts. Exactly like $0C except the script does not end. [0D][xxyy] """ size = 3 byte1 = ord(rom[offset+1]) byte2 = ord(rom[offset+2]) number = byte1 + (byte2 << 8) #0000 to 000AD ... XXX how should these be handled? command["predefined_script_number"] = number elif command_byte == 0x0E: #ASM code1 [3b] info = "ASM code1 [3b]" long_info = """ Calls a predefined routine by interpreting the ASM the pointer points to. [0E][3byte pointer] """ size = 4 command["pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=True) #XXX should we dissassemble the asm at the target location? elif command_byte == 0x0F: #0F codes [xxyy] info = "call some predefined script [0F][yyxx]" long_info = """ Calls predefined scripts. [0F][yyxx] NOTE: For (some) dialogues the font needs to be loaded with the Text box&font code. """ size = 3 byte1 = ord(rom[offset+1]) byte2 = ord(rom[offset+2]) number = byte1 + (byte2 << 8) command["predefined_script_number"] = number elif command_byte == 0x10: #ASM code2 [2b] info = "ASM code2 [2b to 3b to asm]" long_info = """ Call an ASM script via a 2byte pointer pointing to a 3byte pointer. [10][2byte pointer pointing to 3byte pointer pointing to ASM script] """ size = 3 command["pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) #XXX should i include the 3-byte pointer at the target location? #XXX should we dissassemble the asm at the target location? elif command_byte == 0x11: #Trigger event check1 [xxyy] info = "Trigger event check1 [xx][yy]" long_info = """ Check the current number of the trigger event on map (map group/map id). [11][map group][map number] """ size = 3 command["map_group"] = ord(rom[start_address+1]) command["map_id"] = ord(rom[start_address+2]) elif command_byte == 0x12: #Activate trigger event from afar [xxyyzz] info = "Activate trigger event from afar [xx][yy][zz]" long_info = """ Changes trigger event number on map (map bank/map no) to xx. xx = trigger event number that should be activated [12][MapBank][MapNo][xx] """ size = 4 command["map_group"] = ord(rom[start_address+1]) command["map_id"] = ord(rom[start_address+2]) command["trigger_number"] = ord(rom[start_address+3]) elif command_byte == 0x13: #Trigger event check info = "Trigger event check" long_info = """ Checks the number of the trigger events on the current map. [13] """ size = 1 elif command_byte == 0x14: #De-/activate trigger event [xx] info = "De-/activate trigger event [xx]" long_info = """ Changes trigger event number on current map to xx. xx = trigger event number that should be activated [14][xx] deactivate? Just activate a different trigger event number. There's a limit of 1 active trigger. """ size = 2 command["trigger_number"] = ord(rom[start_address+1]) elif command_byte == 0x15: #Load variable into RAM [xx] info = "Load variable into RAM [xx]" long_info = "[15][xx]" size = 2 command["variable"] = ord(rom[start_address+1]) elif command_byte == 0x16: #Add variables [xx] info = "Add variables [xx]" long_info = """ Adds xx and the variable in RAM. #[16][xx] """ size = 2 command["variable"] = ord(rom[start_address+1]) elif command_byte == 0x17: #Random number [xx] info = "Random number [xx]" long_info = """ #Reads xx and creates a random number between 00 and xx -1. #According to this xx can be all but 00. Random number = [00; xx) #The nearer the random number is to xx, the rarer it occurs. #Random number gets written to RAM. """ size = 2 command["rarest"] = ord(rom[start_address+1]) elif command_byte == 0x18: #Version check info = "G/S version check" long_info = """ #Check if version is gold or silver. Gives feedback. #00 = Gold #01 = Silver #[18] """ size = 1 elif command_byte == 0x19: #Copy variable code1 [xxyy] info = "Copy from RAM address to script RAM variable [xxyy]" long_info = """ #Writes variable from ram address to RAM. #[19][2-byte RAM address] """ size = 3 #XXX maybe a pointer is a bad idea? command["pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) elif command_byte == 0x1A: #Copy variable code2 [xxyy] info = "Write variable from script RAM variable to actual RAM address [xxyy]" long_info = """ #Writes variable from RAM to actual RAM address. #[1A][2-byte RAM address] """ size = 3 command["pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) elif command_byte == 0x1B: #Load variable [xxyyzz] info = "Load variable [xxyy][zz]" long_info = """ #Writes zz to ram address. #[1B][2-byte RAM address][zz] """ size = 4 command["pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) command["value"] = ord(rom[start_address+3]) elif command_byte == 0x1C: #Check codes [xx] #XXX no idea what's going on in this one :( info = "Check pre-ID-mapped RAM location [xx]" long_info = """ #Checks special game-technical values and writes then into RAM. #[1C][following part][Ram check (when <> 08/09 see „numbers“ in list of following parts)] #following part (and then hex values) #01 = PKMN count in party # 00 - 06 #02 = ??? #03 = Battle type of wild PKMN #04 = ??? #05 = PokéDex caught # 00 - FA #06 = PokéDex seen # 00 - FA #07 = Badge count # 00 - 10 #08 = Movement # 00 = walk # 01 = bike # 02 = slipping # 04 = surfer # 08 = surfing pikachu #09 = HIRO direction # 00 (d) # 01 (u) # 02 (l) # 03 (r) #0A = Time in hours # 00 - 18 #0B = Day # 00 (Mo) - 06 (Su) #0C = Map bank of current map #0D = Map no of current map #0E = Num. of diff. unowns seen # 00 - 1A #0F = Action byte of map #10 = Amount of free spaces in pkmn box # 00 - 14 #11 = Minutes until end bug contest # 00 - 14 #12 = X position of HIRO #13 = Y position of HIRO #14 = phone call number """ size = 2 #i think? command["following_part"] = ord(rom[start_address+1]) elif command_byte == 0x1D: #Input code1 [xx] info = "Write to pre-ID-mapped RAM location [xx]" long_info = """ #Writes variable from RAM to special game-technical value offsets. #[1D][following part] #where [following part] is the same as 0x1C """ size = 2 command["following_part"] = ord(rom[start_address+1]) elif command_byte == 0x1E: #Input code2 [xxyy] info = "Write byte value to pre-ID-mapped RAM location [aa][xx]" long_info = """ #Writes variable xx to special game-technical value offsets. #[1E][following part][xx] #where [following part] is the same as 0x1C """ size = 3 command["following_part"] = ord(rom[start_address+1]) command["value"] = ord(rom[start_address+2]) elif command_byte == 0x1F: #Give item code [xxyy] info = "Give item by id and quantity [xx][yy]" long_info = """ #Gives item (item no) amount times. #feedback: # 00 = bag full # 01 = OK #[1F][item no][amount] """ size = 3 command["item_id"] = ord(rom[start_address+1]) command["quantity"] = ord(rom[start_address+2]) elif command_byte == 0x20: #Take item code [xxyy] info = "Take item by id and quantity [xx][yy]" long_info = """ #Gives item (item no) amount times #feedback: # 00 = not enough items #[20][item no][amount] """ size = 3 command["item_id"] = ord(rom[start_address+1]) command["quantity"] = ord(rom[start_address+2]) elif command_byte == 0x21: #Check for item code [xx] info = "Check if player has item [xx]" long_info = """ #Checks if item is possessed. #feedback: # 00 = does not have item # 01 = has item #[21][item no] """ size = 2 command["item_id"] = ord(rom[start_address+1]) elif command_byte == 0x22: #Give money code [xxyyzzaa] info = "Give money to HIRO/account [xxyyzzaa]" long_info = """ #Gives zzyyxx money to HIRO/account. #zzyyxx = amount of money (000000 - 0F423F) #[22][00-HIRO/01-account][xxyyzz] """ size = 5 bytes = rom_interval(start_address, size, strings=False) command["account"] = bytes[1] command["amount"] = bytes[2:] #raise NotImplementedError, "don't know if zzyyxx is a decimal or hex value" elif command_byte == 0x23: #Take money code [xxyyzzaa] info = "Take money from HIRO/account [xxyyzzaa]" long_info = """ #Takes zzyyxx money from HIRO/account. #zzyyxx = amount of money (000000 - 0F423F) #[23][00-HIRO/01-account][xxyyzz] """ size = 5 bytes = rom_interval(start_address, size, strings=False) command["account"] = bytes[1] command["amount"] = bytes[2:] #raise NotImplementedError, "don't know if zzyyxx is a decimal or hex value" elif command_byte == 0x24: #Check for money code [xxyyzzaa] info = "Check if HIRO/account has enough money [xxyyzzaa]" long_info = """ #Checks if HIRO/account has got zzyyxx money. #feedback: # 00 = enough money # 01 = exact amount # 02 = less money #zzyyxx = amount of money (000000 - 0F423F) #[24][00-HIRO/01-account][xxyyzz] """ size = 5 bytes = rom_interval(start_address, size, strings=False) command["account"] = bytes[1] command["quantity"] = bytes[2:] #XXX how is quantity formatted? #raise NotImplementedError, "don't know if zzyyxx is a decimal or hex value" elif command_byte == 0x25: #Give coins code [xxyy] info = "Give coins to HIRO [xxyy]" long_info = """ #Gives coins to HIRO. #yyxx = amount of coins (0000 - 270F) #[25][xxyy] """ size = 3 bytes = rom_interval(start_address, size, strings=False) command["quantity"] = bytes[1] + (bytes[2] << 8) elif command_byte == 0x26: #Take coins code [xxyy] info = "Take coins from HIRO [xxyy]" long_info = """ #Takes coins away from HIRO. #yyxx = amount of coins (0000 - 270F) #[26][xxyy] """ size = 3 bytes = rom_interval(start_address, size, strings=False) command["quantity"] = bytes[1] + (bytes[2] << 8) elif command_byte == 0x27: #Check for coins code [xxyy] info = "Check if HIRO has enough coins [xxyy]" long_info = """ #Checks if HIRO has enough coins. #feedback: # 00 = has enough coins # 01 = has exact amount # 02 = does not have enough #yyxx = amount of coins necessary (0000 - 270F) #[27][xxyy] """ size = 3 bytes = rom_interval(start_address, size, strings=False) command["quantity"] = bytes[1] + (bytes[2] << 8) elif command_byte == 0x28: #Give cell phone number [xx] info = "Give cell phone number [xx]" long_info = """ #Gives number to HIRO. #feedback: # 00 = number was added successfully # 01 = Number already added, or no memory #xx = number of person #[28][xx] #01 = mother #02 = bike store #03 = bll #04 = elm """ size = 2 command["number"] = ord(rom[start_address+1]) elif command_byte == 0x29: #Take cell phone number [xx] info = "Delete cell phone number [xx]" long_info = """ #Deletes a number from the list. #feedback: # 00 = number deleted successfully # 01 = number wasn't in list #xx = number of person #[29][xx] """ size = 2 command["number"] = ord(rom[start_address+1]) elif command_byte == 0x2A: #Check for cell phone number [xx] info = "Check for cell phone number [xx]" long_info = """ #Checks if a number is in the list. #feedback: # 00 = number is in list # 01 = number not in list #xx = number to look for #[2A][xx] """ size = 2 command["number"] = ord(rom[start_address+1]) elif command_byte == 0x2B: #Check time of day [xx] info = "Check time of day [xx]" long_info = """ #Checks the time of day. #feedback: # 00 = time of day is the same # 01 = time of day is not the same #[2B][time of day (01morn-04night)] """ size = 2 command["time_of_day"] = ord(rom[start_address+1]) elif command_byte == 0x2C: #Check for PKMN [xx] info = "Check for pokemon [xx]" long_info = """ #Checks if there is a certain PKMN in team. #feedback: # 00 = in team # 01 = not in team #xx = pkmn id #[2C][xx] """ size = 2 command["pokemon_id"] = ord(rom[start_address+1]) elif command_byte == 0x2D: #Give PKMN [xxyyzzaa(+2b +2b)] info = "Give pokemon [pokemon][level][item][trainer2b][...]" long_info = """ #Gives a PKMN if there's space #feedback: # trainer id #[2D][PKMN][PKMNlvl][PKMNitem][TRAINER] #trainer: # 00 = HIRO # 01 = after the main code there are 4 bytes added # [2byte pointer to trainer's name (max.0x0A figures + 0x50)][2byte pointer to nickname (max.0x0A figures + 0x50)] """ size = 5 bytes = rom_interval(start_address, size, strings=False) command["pokemon_id"] = bytes[1] command["pokemon_level"] = bytes[2] command["held_item_id"] = bytes[3] command["trainer"] = bytes[4] if command["trainer"] == 0x01: size += 4 bytes = rom_interval(start_address, size, strings=False) command["trainer_name_pointer"] = calculate_pointer_from_bytes_at(start_address+5, bank=False) command["pokemon_nickname_pointer"] = calculate_pointer_from_bytes_at(start_address+7, bank=False) elif command_byte == 0x2E: #Give EGG [xxyy] info = "Give egg [xx][yy]" long_info = """ #Gives egg if there's space. #feedback: # 00 = OK # 02 = transaction not complete #[2E][PKMN][PKMNlvl] """ size = 3 command["pokemon_id"] = ord(rom[start_address+1]) command["pokemon_level"] = ord(rom[start_address+2]) elif command_byte == 0x2F: #Attach item code [2B] info = "Attach item to last pokemon in list [xxyy]" long_info = """ #Gives last PKMN in list an item and letter text if applicable. Replaces existing items. #[2F][2byte pointer to item no + 0x20 bytes letter text] """ size = 3 command["item_pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) #XXX are those 20 bytes supposed to happen here? or at the pointer's destination? elif command_byte == 0x30: #Check letter code [2b] info = "Check letter against known letter [xxyy]" long_info = """ #Opens a PKMN list. Selected PKMN must have the right letter + the right contents. If OK, then PKMN is taken away #feedback: # 00 = wrong letter # 01 = OK # 02 = Cancelled # 03 = Chosen PKMN has no letter # 04 = Chosen PKMN is the only one in the list. #[30][2byte pointer to letter item no + 0x20 bytes letter text] """ size = 3 command["item_pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) elif command_byte == 0x31: #BitTable1 check [xxyy] info = "Check some bit on bit table 1 [xxyy]" long_info = """ #Checks whether a bit of BitTable1 has the value 0 or 1. #feedback: # 00 = value 0 (off) # 01 = value 1 (on) #[31][2-byte bit number] """ #XXX what format is the 2-byte number in? size = 3 bytes = rom_interval(start_address+1, size-1, strings=False) command["bit_number_bytes"] = bytes #raise NotImplementedError, "what format is the 2-byte number in?" elif command_byte == 0x32: #BitTable1 reset [xxyy] info = "Reset (to 0) a bit on bit table 1 [xxyy]" long_info = """ #Sets a bit of BitTable1 to value 0. #[32][Bit no (2byte)] """ size = 3 bytes = rom_interval(start_address+1, size-1, strings=False) command["bit_number_bytes"] = bytes elif command_byte == 0x33: #BitTable1 set [xxyy] info = "Set (to 1) a bit on bit table 1 [xxyy]" long_info = """ #Sets a bit of BitTable1 to value 1. #[33][Bit-No (2byte)] """ size = 3 bytes = rom_interval(start_address+1, size-1, strings=False) command["bit_number_bytes"] = bytes elif command_byte == 0x34: #BitTable2 check [xxyy] info = "Check some bit on bit table 2 [xxyy]" long_info = """ #Checks whether a bit of BitTable2 has the value 0 or 1. #feedback: # 00 = value 0 (off) # 01 = value 1 (on) #[34][Bit no (2byte)] """ size = 3 bytes = rom_interval(start_address+1, size-1, strings=False) command["bit_number_bytes"] = bytes elif command_byte == 0x35: #BitTable2 reset [xxyy] info = "Reset (to 0) a bit on bit table 2 [xxyy]" long_info = """ #Sets a bit of BitTable2 to value 0. #[35][Bit no (2byte)] """ size = 3 bytes = rom_interval(start_address+1, size-1, strings=False) command["bit_number_bytes"] = bytes elif command_byte == 0x36: #BitTable2 set [xxyy] info = "Set (to 1) a bit on bit table 2 [xxyy]" long_info = """ #Sets a bit of BitTable2 to value 1. #[36][Bit no (2byte)] """ size = 3 bytes = rom_interval(start_address+1, size-1, strings=False) command["bit_number_bytes"] = bytes elif command_byte == 0x37: #Deactivate PKMN battles info = "Turn off wild pokemon battles" long_info = """ #This code turns all wild PKMN battles off. #[37] """ size = 1 elif command_byte == 0x38: #Activate PKMN battles info = "Turn no wild pokemon battles" long_info = "This code turns all wild PKMN battles on." size = 1 elif command_byte == 0x39: #X/Y comparison [xxyy] info = "BUGGY x/y comparison [xxyy]" long_info = """ #This code is buggy (Bug fix: 0x3021 --> C6) and can't used as #described without fix. This code compares the X and Y coordinates of #HIRO with the ones in a table (max. 20h XY pairs) on the current map. #It sets or resets the 4 bytes D17C to D17F accordingly to this table, #1 bit for every table entry. To be useful, this code can only be used #in a command queue, because with every regular move of HIRO the bits #are reset again. This code is an alternative to the trigger events and #can be used via the command queue code. #See Write command queue, Additional documentation: 3:4661 with c= index #in table (start=00), hl=D171, b=01, d=00. """ size = 3 command["table_pointer"] = rom_interval(start_address+1, size-1, strings=False) elif command_byte == 0x3A: #Warp modifier [xxyyzz] info = "Set target for 0xFF warps [warp id][map group][map id]" long_info = """ #Changes warp data for all warps of the current map that have a 0xFF for warp-to data. #[3A][Nee warp-to][New map bank][New map no] """ size = 4 bytes = rom_interval(start_address+1, size-1, strings=False) command["nee_warp_to"] = bytes[0] command["map_group"] = bytes[1] command["map_id"] = bytes[2] elif command_byte == 0x3B: #Blackout modifier [xxyy] info = "Blackout warp modifier [map group][map id]" long_info = """ #Changes the map HIRO arrives at, after having a blackout. #There needs to be flying data for that map. #[3B][Map bank][Map no] """ size = 3 command["map_group"] = ord(rom[start_address+1]) command["map_id"] = ord(rom[start_address+2]) elif command_byte == 0x3C: #Warp code [xxyyzzaa] info = "Warp to [map group][map id][x][y]" long_info = """ #Warps to another map. #If all data is 00s, then the current map is reloaded with #the current X and Y coordinates. Old script is not finished #without a [90]. #[3C][Map bank][Map no][X][Y] """ size = 5 command["map_group"] = ord(rom[start_address+1]) command["map_id"] = ord(rom[start_address+2]) command["x"] = ord(rom[start_address+3]) command["y"] = ord(rom[start_address+4]) elif command_byte == 0x3D: #Account code [xxyy] info = "Read money amount [xx][yy]" long_info = """ #Reads amount of money in accounts of HIRO and mother and writes #it to MEMORY1, 2 or 3 for later use in text. #[3D][00 = HIRO| <> 00 = Mother][00-02 MEMORY] #see http://hax.iimarck.us/files/scriptingcodes_eng.htm#InText01 """ size = 3 command["account_id"] = ord(rom[start_address+1]) command["memory_id"] = ord(rom[start_address+2]) elif command_byte == 0x3E: #Coin case code [xx] info = "Read coins amount [xx]" long_info = """ #Reads amount of coins in coin case and writes it to MEMORY 1, 2, #or 3 for later use in text. #[3E][00-02 MEMORY] """ size = 2 command["memory_id"] = ord(rom[start_address+1]) elif command_byte == 0x3F: #Display RAM [xx] info = "Copy script RAM value into memX [xx]" long_info = """ #Reads RAM value and writes it to MEMORY1, 2 or 3 for later use in text. #[3F][00-02 MEMORY] """ size = 2 command["memory_id"] = ord(rom[start_address+1]) elif command_byte == 0x40: #Display pokémon name [xxyy] info = "Copy pokemon name (by id) to memX [id][xx]" long_info = """ #Writes pokémon name to MEMORY1, 2 or 3 for later use in text. #[40][PKMN no][00-02 MEMORY] """ size = 3 command["map_id"] = ord(rom[start_address+1]) command["memory_id"] = ord(rom[start_address+2]) elif command_byte == 0x41: #Display item name [xxyy] info = "Copy item name (by id) to memX [id][xx]" long_info = """ #Writes item name to MEMORY1, 2 or 3 for later use in text. #[41][Item no][00-02 MEMORY] """ size = 3 command["item_id"] = ord(rom[start_address+1]) command["memory_id"] = ord(rom[start_address+2]) elif command_byte == 0x42: #Display location name [xx] info = "Copy map name to memX [xx]" long_info = """ #Writes current location's name to MEMORY1, 2 or 3 for later use in text. #[42][00-02 MEMORY] """ size = 2 command["memory_id"] = ord(rom[start_address+1]) elif command_byte == 0x43: #Display trainer name [xxyyzz] info = "Copy trainer name (by id&group) to memZ [xx][yy][zz]" long_info = """ #Writes trainer name to MEMORY1, 2 or 3 for later use in text. #[43][Trainer number][Trainer group][00-02 MEMORY] """ size = 4 command["trainer_id"] = ord(rom[start_address+1]) command["trainer_group"] = ord(rom[start_address+2]) command["memory_id"] = ord(rom[start_address+3]) elif command_byte == 0x44: #Display strings [2b + xx] info = "Copy text (by pointer) to memX [aabb][xx]" long_info = """ #Writes string to MEMORY1, 2 or 3 for later use in text. #[44][2byte pointer to string (max. 0x0C figures + 0x50)][00-02 MEMORY] #See 0C codes: 0C2900, 0C2A00, 0C1B00, 0C2200, Usage of variable strings in text. """ size = 4 command["pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) command["memory_id"] = ord(rom[start_address+3]) command["text"] = parse_text_engine_script_at(command["pointer"], map_group=map_group, map_id=map_id, debug=debug) elif command_byte == 0x45: #Stow away item code info = "Show HIRO put the ITEMNAME in the ITEMPOCKET text box" long_info = """ #Text box: "HIRO put the ITEMNAME in the ITEMPOCKET." #The item number has to have been loaded beforehand #(e.g. by Give item code). """ size = 1 elif command_byte == 0x46: #Full item pocket code info = "Show ITEMPOCKET is full textbox" long_info = """ #Text box: "ITEMPOCKET is full..." The item number has to have #been loaded beforehand (e.g. by Give item code). """ size = 1 elif command_byte == 0x47: #Text box&font code info = "Loads the font into the ram and opens a text box." size = 1 elif command_byte == 0x48: #Refresh code [xx] info = "Screen refresh [xx]" long_info = """ #Executes a complete screen refresh. #[48][xx] #xx is a dummy byte """ size = 2 command["dummy"] = ord(rom[start_address+1]) elif command_byte == 0x49: #Load moving sprites info = "Load moving sprites into memory" long_info = "Loads moving sprites for person events into ram." size = 1 elif command_byte == 0x4A: #Load byte to C1CE [xx] info = "Load specific byte to $C1CE [xx]" long_info = """ #Loads a byte to C1CE. Seems to have no function in the game. #[4A][Byte] """ size = 2 command["byte"] = ord(rom[start_address+1]) elif command_byte == 0x4B: #Display text [3b] info = "Display text by pointer [bb][xxyy]" long_info = """ #Opens a text box and writes text. Doesn't load font. #[4B][Text bank][2byte text pointer] """ size = 4 command["text_group"] = ord(rom[start_address+1]) command["pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=True) command["text"] = parse_text_engine_script_at(command["pointer"], map_group=map_group, map_id=map_id, debug=debug) elif command_byte == 0x4C: #Display text [2b] info = "Display text by pointer [xxyy]" long_info = """ #Opens a text box and writes text. Doesn't load font. #[4C][2byte text pointer] """ size = 3 command["pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) command["text"] = parse_text_engine_script_at(command["pointer"], map_group=map_group, map_id=map_id, debug=debug) elif command_byte == 0x4D: #Repeat text [xxyy] info = "Repeat text [FF][FF]" long_info = """ #Opens a text box and writes the text written latest resp. whose address was put statically to D175-D177. #Doesn't load font. #[4D][FF][FF] #Without FF for both bytes, no operation occurs """ size = 3 command["bytes"] = rom_interval(start_address+1, 2, strings=False) elif command_byte == 0x4E: #YES/No box info = "YES/No box" long_info = """ #Displays a YES/NO box at X0F/Y07 #feedback: # 00 = no # 01 = yes """ size = 1 elif command_byte == 0x4F: #Menu data code [2b] info = "Load menu data by pointer [xxyy]" long_info = """ #Loads data for menus #[4F][2byte pointer to menu data] #see http://hax.iimarck.us/files/scriptingcodes_eng.htm#ZusatzDatA4F """ size = 3 command["menu_pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) elif command_byte == 0x50: #Write backup code info = "Write screen backup" long_info = "Writes backup of parts of the screen the box was overlapping." size = 1 elif command_byte == 0x51: #Text1 code [2b] info = "Display text (by pointer), turn to HIRO, end [xxyy]" long_info = """ #Displays a text and lets person turn to HIRO. #Afterwards there is no other script interpreted. #Corresponds to 6A + 47 + 4C + 53 + 49 + 90 #[51][2byte textpointer] """ size = 3 end = True command["pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) command["text"] = parse_text_engine_script_at(command["pointer"], map_group=map_id, map_id=map_id, debug=debug) elif command_byte == 0x53: #Text2 code [2b] info = "Display text (by pointer) and end [xxyy]" long_info = """ #Displays a text. Afterwards there is no other script interpreted. #Corresponds to 47 + 4C + 53 + 49 + 90 #[52][2byte textpointer] """ size = 3 end = True command["pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) command["text"] = parse_text_engine_script_at(command["pointer"], map_group=map_id, map_id=map_id, debug=debug) elif command_byte == 0x54: #Close text box code info = "Close text box" long_info = "Closes a text box which was opened by 47 resp. 4B/4C/4D." size = 1 elif command_byte == 0x55: #Keep text box open code info = "Keep text box open" long_info = "Keeps a text box open which was opened by 47 resp. 4B/4C/4D." size = 1 elif command_byte == 0x56: #Pokémon picture code [xx] info = "Display a pokemon picture in a box by pokemon id [xx]" long_info = """ #Opens a box and puts a Pokémon picture into it. #[55][xx] #xx: # <>00 : Pokémon no # =00 : Pokémon no gets read from RAM """ size = 2 command["byte"] = ord(rom[start_address+1]) elif command_byte == 0x57: #Pokémon picture YES/NO code info = "?? Display a pokemon picture and a yes/no box" long_info = """ #Displays a YES/NO box at X08/Y05. #feedback: # 00 = no chosen # 01 = yes chosen """ size = 1 elif command_byte == 0x58: #Menu interpreter 1 info = "Menu interpreter 1 (see menu loader)" long_info = """ #Interprets menu data loaded by 4F. #see also http://hax.iimarck.us/files/scriptingcodes_eng.htm#ZusatzDatA57 """ size = 1 elif command_byte == 0x59: #Menu interpreter 2 info = "Menu interpreter 2 (see menu loader)" long_info = """ #Interprets menu data loaded by 4F. #see also http://hax.iimarck.us/files/scriptingcodes_eng.htm#Marke57 #see also http://hax.iimarck.us/files/scriptingcodes_eng.htm#ZusatzDatA58 """ size = 1 elif command_byte == 0x5A: #Load Pikachu data info = "Load pikachu data" long_info = "Loads 0x19 (Pikachu) to PokéRAM and level 5 to LevelRAM." size = 1 elif command_byte == 0x5B: #Delete FightRAM/reset person check info = "? Disable fleeing from battle" long_info = """ #Deletes the value in BattleRAM. #Turns off the check if the battle was started by entering #a trainer's area of view. """ size = 1 elif command_byte == 0x5C: #Load trainer data1 info = "Load trainer from RAM" long_info = """ #Loads trainer data when HIRO is in a trainer's range of sight. #Trainer group is read from CF2E and written to #TrRAM1, the trainer number is read from CF2F and written to #TrRAM2. 81 is written to BattleRAM. """ size = 1 elif command_byte == 0x5D: #Load Pokémon data [xxyy] info = "Loads pokemon by id and level for BattleRAM [xx][yy]" long_info = """ #Loads Pokémon data. Writes 80 to BattleRAM. #[5C][Poke no][Level] """ size = 3 command["pokemon_id"] = ord(rom[start_address+1]) command["pokemon_level"] = ord(rom[start_address+2]) elif command_byte == 0x5E: #Load trainer data2 [xxyy] info = "Load trainer by group/id for BattleRAM [xx][yy]" long_info = """ #Loads trainer data. Trainer group --> TrRAM1, #trainer number --> TrRAM2. Writes 81 to BattleRAM. #[5D][Trainer group][Trainer no] """ size = 3 command["trainer_group"] = ord(rom[start_address+1]) command["trainer_id"] = ord(rom[start_address+2]) elif command_byte == 0x5F: #Start battle info = "Start pre-configured battle" long_info = """ #Starts trainer or Pokémon battle. BattleRAM: 80 = Poké battle; 81 = Trainer battle. #feedback: # 00 = win # 01 = lose """ size = 1 elif command_byte == 0x60: #Return to In game engine after battle info = "Return to in-game engine after battle" long_info = "Returns to ingame engine and evaluates battle. When lost then return to last Pokémon Center etc." size = 1 elif command_byte == 0x61: #Learn how to catch PKMN [xx] info = "Pokemon catching tutorial [xx]" long_info = """ #Starts a learn-how-to-catch battle with a Pokémon, whose data needs to be loaded beforehand #see http://hax.iimarck.us/files/scriptingcodes_eng.htm#Marke5C #Player has to have at least 1 Pokémon for it to work. #Items that are statically used: 1xPotion, 5xPoké ball. #[60][xx] #xx: Between 01 and 03. If <> 03 then HIRO sprite instead of dude sprite and kills #itself when using the item system. """ size = 2 command["byte"] = ord(rom[start_address+1]) elif command_byte == 0x62: #Trainer text code info = "Set trainer text by id [xx]" long_info = """ #Interprets the data of a in the event structure defined trainer. #[61][xx] #Xx decides which text to use. #xx: Between 00 and 03. #see http://hax.iimarck.us/files/scriptingcodes_eng.htm#Eventaufbau """ size = 2 command["byte"] = ord(rom[start_address+1]) elif command_byte == 0x63: #Trainer status code [xx] info = "? Check trainer status [xx]" long_info = """ #Checks/changes the status of a in the event structure defined trainer. #see http://hax.iimarck.us/files/scriptingcodes_eng.htm#Eventaufbau #[62][xx] #xx is: # 00 = deactivate # 01 = activate # 02 = check """ size = 2 command["byte"] = ord(rom[start_address+1]) elif command_byte == 0x64: #Pointer Win/Lose [2b + 2b] info = "Set win/lose pointers for battle [xxyy][xxyy]" long_info = """ #Writes the win/lose pointer of a battle into the ram. #[63][2byte pointer to text Win][2byte pointer to text Loss*] #* When pointer = 0000 then "Blackout" instead of return to gameplay. """ size = 5 #sometimes win/lost can be a pointer to 0000 or None? command["won_pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) command["lost_pointer"] = calculate_pointer_from_bytes_at(start_address+3, bank=False) if command["won_pointer"] == None: command["won_pointer"] = 0 else: command["text_won"] = parse_text_engine_script_at(command["won_pointer"], map_group=map_id, map_id=map_id, debug=debug) if command["lost_pointer"] == None: command["lost_pointer"] = 0 else: command["text_lost"] = parse_text_engine_script_at(command["lost_pointer"], map_group=map_id, map_id=map_id, debug=debug) elif command_byte == 0x65: #Script talk-after #XXX this is a really poor description of whatever this is info = "? Load the trainer talk-after script" long_info = """ #Interprets which script is going to be run, when a in the event-structure-defined #trainer is talked to again. #see http://hax.iimarck.us/files/scriptingcodes_eng.htm#Eventaufbau #[64] """ size = 1 elif command_byte == 0x66: #Script talk-after-cancel info = "Disable/cancel trainer after-battle text" long_info = """ #Cancels the talk-after script of the in the event-structure-defined #trainer when talk-after script is executed just after the battle. #[65] """ size = 1 elif command_byte == 0x67: #Script talk-after-check #XXX also bad explanation/name... info = "? Check if trainer talk-after script is executed just after battle or not" long_info = """ #Checks if the talk-after script of the event structure defined trainer #is executed just after the battle or at a later point in time. #feedback: # 00 = no # 01 = yes #[66] """ size = 1 elif command_byte == 0x68: #Set talked-to person [xx] info = "Set last talked-to person [xx]" long_info = """ #Sets the number of the last person talked to. #[67][person] """ size = 2 command["person_id"] = ord(rom[start_address+1]) elif command_byte == 0x69: #Moving code [xx + 2b] info = "Move person (by id) with moving data (by pointer) [id][xxyy]" long_info = """ #Moves the person using moving data. #[68][Person][2byte pointer to moving data] #see also http://hax.iimarck.us/files/scriptingcodes_eng.htm#ZusatzB68bis69 """ size = 4 command["person_id"] = ord(rom[start_address+1]) command["moving_data_pointer"] = calculate_pointer_from_bytes_at(start_address+2, bank=False) elif command_byte == 0x6A: #Moving code for talked-to person [2b] info = "Move talked-to person with moving data (by pointer) [xxyy]" long_info = """ #Moves talked-to person using moving data. #[69][2byte pointer to moving data] """ size = 3 command["moving_data_pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) elif command_byte == 0x6B: #Talk-to facing code info = "Move talked-to person's facing direction to HIRO" long_info = """ #Turns the heads of the talked-to persons to HIRO. #[6A] """ size = 1 elif command_byte == 0x6C: #Facing of people code [xxyy] info = "Move facing direction of person1 to look at person2 [2][1]" long_info = """ #Turns the head of person1 to another person2. #[6B][Person2][Person1] #Person2 = If number is greater than 0xFD, then use number of talked-to person. #Person1 = If number equals 0xFE, then take number of talked-to person. """ size = 3 command["person2_id"] = ord(rom[start_address+1]) command["person1_id"] = ord(rom[start_address+2]) elif command_byte == 0x6D: #Variable sprites [xxyy] info = "Store value in variable sprite RAM location x by id Y [xx][yy]" long_info = """ #Writes a number to the variable sprite RAM from D555 to D564 (see Compendium on the sprite system). #[6C][xx][Sprite no] #xx: Number between 0x00 and 0x0F """ size = 3 command["number"] = ord(rom[start_address+1]) command["sprite_id"] = ord(rom[start_address+2]) elif command_byte == 0x6E: #Hide person [xx] info = "Hide person by id [xx]" long_info = """ #Hides a person. #[6D][person id] """ size = 2 command["person_id"] = ord(rom[start_address+1]) elif command_byte == 0x6F: #Show person [xx] info = "Show person by id [xx]" long_info = """ #Shows a hidden person again. #[6E][person id] """ size = 2 command["person_id"] = ord(rom[start_address+1]) elif command_byte == 0x70: #Following code1 [xxyy] info = "Following code1 [leader id][follower id]" long_info = """ #A person1 follows another person2. The person1 that follows #just repeats the movement of person2, even if the persons are #not directly next to each other. #[6F][Leader Person2][Follower Person1] """ size = 3 command["leader_person_id"] = ord(rom[start_address+1]) command["follower_person_id"] = ord(rom[start_address+2]) elif command_byte == 0x71: #Stop following code info = "Stop all follow code" long_info = "Ends all current follow codes." size = 1 elif command_byte == 0x72: #Move person [xxyyzz] info = "Move person by id to xy [id][xx][yy]" long_info = """ #Sets the X/Y values of a person anew. #The person doesn't get shown immediately. Use hide&show. #[71][Person][X][Y] """ size = 4 command["person_id"] = ord(rom[start_address+1]) command["x"] = ord(rom[start_address+2]) command["y"] = ord(rom[start_address+3]) elif command_byte == 0x73: #Write person location [xx] (lock person location?) info = "Lock person's location by id [id]" long_info = """ #Writes the current X/Y values of a person into the ram. #The person is going to stand at its current location even when #it's out of HIRO's sight and is not going to return to its old #location until the next map load. #[72][person] """ size = 2 command["person_id"] = ord(rom[start_address+1]) elif command_byte == 0x74: #Load emoticons [xx] info = "Load emoticon bubble [xx]" long_info = """ #Loads the emoticon bubble depending on the given bubble number. #[73][bubble number] #xx: If xx = FF then take number from RAM. # 00 = Exclamation mark # 01 = Question mark # 02 = Happy # 03 = Sad # 04 = Heart # 05 = Flash # 06 = Snoring # 07 = Fish """ size = 2 command["bubble_number"] = ord(rom[start_address+1]) elif command_byte == 0x75: #Display emoticon [xxyyzz] info = "Display emoticon by bubble id and person id and time [xx][yy][zz]" long_info = """ #Displays the bubble above a persons head for the given time period. #Attention: Bubbles get loaded into ram! #[74][Bubble][Person][Time] #for bubble ids see 0x73 """ size = 4 command["bubble_number"] = ord(rom[start_address+1]) command["person_id"] = ord(rom[start_address+2]) command["time"] = ord(rom[start_address+3]) elif command_byte == 0x76: #Change facing [xxyy] info = "Set facing direction of person [person][facing]" long_info = """ #Changes the facing direction of a person. #[75][person][facing] """ size = 3 command["person_id"] = ord(rom[start_address+1]) command["facing"] = ord(rom[start_address+2]) elif command_byte == 0x77: #Following code2 [xxyy] info = "Following code2 [leader id][follower id]" long_info = """ #A person1 follows a person2. The following person1 automatically clings to person2. #Person1 just follows person2, but doesn't make the exact same movements at person2. #[76][Leader Person2][Follower Person1] """ size = 3 command["leader_person_id"] = ord(rom[start_address+1]) command["follower_person_id"] = ord(rom[start_address+2]) elif command_byte == 0x78: #Earth quake [xx] info = "Earthquake [xx]" long_info = """ #The screen shakes. xx gives time as well as displacement of the screen. #[77][xx] """ size = 2 command["shake_byte"] = ord(rom[start_address+1]) elif command_byte == 0x79: #Exchange map [3b] info = "Draw map data over current map [bank][pointer]" long_info = """ #This code draws another whole map as wide and high as the #current map over the current map. #The 3byte pointer points to the new map. #[78][3byte pointer to new map data] """ size = 4 command["map_data_pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=True) elif command_byte == 0x7A: #Change block code [xxyyzz] info = "Change block to block id on map [xx][yy][id]" long_info = """ #Changes a block on the current map by giving the new block #number and its X/Y values measured in half-blocks. #[79][X][Y][Block] """ size = 4 command["x"] = ord(rom[start_address+1]) command["y"] = ord(rom[start_address+2]) command["block"] = ord(rom[start_address+3]) elif command_byte == 0x7B: #Reload map code info = "Reload/redisplay map" long_info = """ #Reloads and re-displays the map completely. #Loads tileset and all map data anew. Screen gets light. #[7A] """ size = 1 elif command_byte == 0x7C: #Reload map part code info = "Reload/redisplay map portion occupied by HIRO" long_info = """ #Reloads and re-displays the part of the map HIRO is on, #without reloading any other map data or the tileset. #[7B] """ size = 1 elif command_byte == 0x7D: #Write command queue info = "Write command queue [xxyy]" long_info = """ #Writes a command queue to the next free slot in ram. #Max 4 command queues à 5 bytes. This code is buggy (bug fix: 25:7C74 --> 12). #[7C][2byte pointer to 5byte command queue] #see http://hax.iimarck.us/files/scriptingcodes_eng.htm#ZusatzDok25_7CC9 """ size = 3 command["command_queue_pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) elif command_byte == 0x7E: #Delete command queue info = "Delete command queue" long_info = """ #Deletes a command queue and frees a slot in ram. #[7D][First command of the resp. queue] """ #XXX wtf? size = 2 command["first_command"] = ord(rom[start_address+1]) elif command_byte == 0x7F: #Song code1 [xxyy] info = "Play music by number [xxyy]" long_info = """ #Immediately plays the music. #[7E][Music no (2byte)] #Music no: See the music archive that should accompany #this document Thanks to Filb. He dumped all the songs #via gameboy player and gave them to me. """ size = 3 #XXX what is the format of this music data? command["music_number"] = rom_interval(start_address+1, size-1, strings=False) elif command_byte == 0x80: #Song code2 info = "Song code2" long_info = """ #Plays the music of the trainer group in TrRAM1. #Takes music numbers from list at 3A:5027. #[7F] """ size = 1 elif command_byte == 0x81: #Music fade-out code [xxyy][zz] info = "Music fade-out then play next [xxyy][time]" long_info = """ #The current music is faded out and the new music is played afterwards. #[80][Music no (2byte)][Time to fade out (00-7F)] """ size = 4 command["music_number"] = rom_interval(start_address+1, 2, strings=False) command["fade_time"] = ord(rom[start_address+3]) elif command_byte == 0x82: #Play map music code info = "Play map's music" long_info = """ #Starts playing the original map music. #Includes special check for surfer and bug contest song. #[81] """ size = 1 elif command_byte == 0x83: #Map reload music code info = "Reload map music" long_info = """ #After a map reload no music is played. #[82] """ size = 1 elif command_byte == 0x84: #Cry code [xx00] info = "Play cry by id or RAM [cry][00]" long_info = """ #Plays the Pokémon's cry. #[83][Cry no][00] #If the cry no = 00 then the number is taken from RAM. """ size = 3 command["cry_number"] = ord(rom[start_address+1]) command["other_byte"] = ord(rom[start_address+2]) elif command_byte == 0x85: #Sound code [xxyy] info = "Play sound by sound number [xxyy]" long_info = """ #Plays the sound. #[84][Sound no (2byte)] #Sound no: See the music archive that should accompany this document #Thanks to philb for this matter. He helped me to record a big #part of these sounds. """ size = 3 command["sound_number"] = rom_interval(start_address+1, 2, strings=False) elif command_byte == 0x86: #Key-down code info = "Wait for key-down" long_info = """ #Waits for the Player to press a button. #[85] """ size = 1 elif command_byte == 0x87: #Warp sound info = "Warp sound" long_info = """ #Evaluates which sound is played when HIRO enters a Warp field. #Usage via script ingame is rather not useful. #[86] """ size = 1 elif command_byte == 0x88: #Special sound info = "Special sound if TM was last checked" long_info = """ #When last given/checked Item was a TM then it plays sound 0x9B. If not, then 0x01. #[87] """ size = 1 elif command_byte == 0x89: #Engine remote control [2b] info = "Engine remote control [bb][xxyy]" long_info = """ #This code controls the engine via "data stream". #[88][3byte pointer to control structure] #see http://hax.iimarck.us/files/scriptingcodes_eng.htm#ZusatzDatA88 """ size = 4 command["data_stream_pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=True) elif command_byte == 0x8A: #Load map anew [xx] info = "Load map with specific loading process [xx]" long_info = """ #The number decides which map loading process is used. #The number must be 0xF0 + process number to work correctly. #[89][Number] #see map loading process: # http://hax.iimarck.us/files/scriptingcodes_eng.htm#ZusatzDok5_5550 """ size = 2 command["number"] = ord(rom[start_address+1]) elif command_byte == 0x8B: #Waiting code [xx] info = "Wait code" long_info = """ #This code lets the game wait for 2 * xx time intervals. #[8A][xx] #xx: Numbers from 0x01 to 0xFF. #If 0x00 is chosen then the time can be manipulated by previously loading a number to RAM2. """ size = 2 command["time"] = ord(rom[start_address+1]) elif command_byte == 0x8C: #Deactivate static facing [xx] info = "Deactive static facing after time [xx]" long_info = """ #Deactivates static facings on all persons on the screen after a time xx. #[8B][xx] """ size = 2 command["time"] = ord(rom[start_address+1]) elif command_byte == 0x8D: #Priority jump1 [2b] info = "Priority jump to script by pointer [xxyy]" long_info = """ #The pointer acts like code 00, but with this higher #functions like the bike etc. are not paid attention to, #while the script is running. #[8C][2byte pointer to script] """ size = 3 script_pointer = calculate_pointer_from_bytes_at(start_address+1, bank=False) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(script_pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(script_pointer, original_start_address, debug=debug) command["script_pointer"] = script_pointer command["script"] = script end = True #according to pksv elif command_byte == 0x8E: #Warp check info = "Reactive all engine checks if player is warping" long_info = """ #If HIRO is entering or leaving a warp then this code reactivates all the engine-checks. #[8D] """ size = 1 elif command_byte == 0x8F: #Priority jump2 [2b] info = "Priority jump to script by pointer (after 1st cycle) [xxyy]" long_info = """ #The pointer acts like code 03, but with this code all #higher functions wait for a cycle before the script gets interpreted. #[8E][2byte pointer to script] """ size = 3 script_pointer = calculate_pointer_from_bytes_at(start_address+1, bank=False) if debug: print "in script starting at "+hex(original_start_address)+\ " about to parse script at "+hex(script_pointer)+\ " called by "+info+" byte="+hex(command_byte) script = rec_parse_script_engine_script_at(script_pointer, original_start_address, debug=debug) command["script_pointer"] = script_pointer command["script"] = script end = True #according to pksv elif command_byte == 0x90: #Return code1 info = "Return code 1" long_info = """ #Ends the current script and loads the backup offset for "linked" #scripts if applicable. The sophisticated functions are not affected #and run like before the code. This code is mostly used for scripts #called by the 2nd part of the script header, because else malfunctions #occur. #[8F] """ size = 1 end = True elif command_byte == 0x91: #Return code2 info = "Return code 2" long_info = """ #Ends the current script and loads the backup offset for "linked" #scripts if applicable. The sophisticated functions get reset if #no backup offset was loaded. This code is used to end most scripts. #[90] """ size = 1 end = True elif command_byte == 0x92: #Return code3 info = "Return code 3" long_info = """ #Reloads the map completely like the code 0x7A #and else acts completely like Return code2 #[91] #see reload map code # http://hax.iimarck.us/files/scriptingcodes_eng.htm#Marke7A #see 0x90 """ size = 1 #XXX does this end the script?? "else acts like 0x90" # else? what's the "if"? end = True elif command_byte == 0x93: #Reset sophisticated functions info = "Reset sophisticated functions" long_info = """ #Resets all sophisticated functions to 0. #[92] """ size = 1 elif command_byte == 0x94: #Mart menu [xxyyzz] info = "Mart menu [dialog no][mart no 2b]" long_info = """ #Displays a whole mart menu, however, doesn't load font to ram. #[93][Dialog no][Mart no (2byte)] #see http://hax.iimarck.us/files/scriptingcodes_eng.htm#AwBsp93 #see http://hax.iimarck.us/files/scriptingcodes_eng.htm#ZusatzB93 """ size = 4 command["dialog_number"] = ord(rom[start_address+1]) command["mart_number"] = rom_interval(start_address+2, 2, strings=False) elif command_byte == 0x95: #Elevator menu [2b] info = "Display elevator menu by pointer [xxyy]" long_info = """ #Displays a whole elevator menu, but it doesn't load font to ram. #Only works with warps with warp-to = 0xFF. #[94][2byte pointer to floor list] """ size = 3 command["floor_list_pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) elif command_byte == 0x96: #Trade menu [xx] info = "Display trade menu by trade id [xx]" long_info = """ #Displays a whole trade menu, but it doesn't load font to ram. #[95][trade no] #see http://hax.iimarck.us/files/scriptingcodes_eng.htm#ZusatzDokTausch """ size = 2 command["trade_number"] = ord(rom[start_address+1]) elif command_byte == 0x97: #Give cell phone number with YES/NO [xx] info = "Give cell phone number by id with YES/NO [id]" long_info = """ #Gives a telephone number but asks for decision beforehand. #feedback: # 00 = ok chosen # 01 = Cell phone number already registered/Memory full # 02 = no chosen #[96][Cell phone number] """ size = 2 #maybe this next param should be called "phone_number" command["number"] = ord(rom[start_address+1]) elif command_byte == 0x98: #Call code [2b] info = "Call code pointing to name of caller [xxyy]" long_info = """ #Displays the upper cell phone box and displays a freely selectable name. #[97][2byte pointer to name of caller] """ size = 3 command["caller_name_pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank=False) elif command_byte == 0x99: #Hang-up code info = "Hang-up phone" long_info = """ #Simulates the hanging-up. #[98] """ size = 1 elif command_byte == 0x9A: #Decoration code [xx] info = "Set monologue decoration [xx]" long_info = """ #Displays monologues according to the selected ornament. #[99][xx] #xx values: # 00 = Map/Poster # 01 = Ornament left # 02 = Ornament right # 03 = Huge doll # 04 = Console """ size = 2 command["ornament"] = ord(rom[start_address+1]) elif command_byte == 0x9B: #Berry tree code [xx] info = "Berry tree by tree id [xx]" long_info = """ #Creates a typical berry tree monologue. #There is a maximum of 32 berry trees in the game. #After this code the script ends. #[9A][Fruit tree number] #Fruit tree number + 11:4091 is the offset where the item no of the berry is defined. """ size = 2 end = True command["tree_id"] = ord(rom[start_address+1]) elif command_byte == 0x9C: #Cell phone call code [xx00] #XXX confirm this? info = "Cell phone call [2-byte call id]" #was originally: [call id][00] long_info = """ #Initiates with the next step on a outer world map (permission byte) a phone call. #[9B][Call no] and maybe [00] ??? #call no: # 01 = PokéRus # 02 = Pokémon stolen # 03 = Egg examined/ Assistant in Viola City # 04 = Team Rocket on the radio # 05 = PROF. ELM has got something for HIRO # 06 = Bike shop gives bike away # 07 = Mother is unhappy that HIRO didn't talk to her before leaving # 08 = PROF. ELM has got something for HIRO a second time """ size = 3 command["call_id"] = ord(rom[start_address+1]) command["id"] = rom_interval(start_address+2, 2, strings=False) elif command_byte == 0x9D: #Check cell phone call code info = "Check if/which a phone call is active" long_info = """ #Checks if a phone call is "in the line". #feedback: # 00 = no # <>00 = call number #[9C] """ size = 1 elif command_byte == 0x9E: #Commented give item code [xxyy] info = "Give item by id and quantity with 'put in pocket' text [id][qty]" long_info = """ #The same as 0x1F but this code comments where #HIRO puts what item in a short monologue. #[9D][Item][Amount] """ size = 3 command["item_id"] = ord(rom[start_address+1]) command["quantity"] = ord(rom[start_address+2]) elif command_byte == 0x9F: #Commented ive item code? info = "Give item by id and quantity with 'put in pocket' text [id][qty]" long_info = """ #The same as 0x1F but this code comments where #HIRO puts what item in a short monologue. #[9D][Item][Amount] """ size = 3 command["item_id"] = ord(rom[start_address+1]) command["quantity"] = ord(rom[start_address+2]) elif command_byte == 0xA0: #Load special wild PKMN data [xxyy] info = "Load wild pokemon data for a remote map [map group][map id]" long_info = """ #Activates the checks in the special tables for the wild pokémon data. #[9E][map group][map id] #see also http://hax.iimarck.us/files/scriptingcodes_eng.htm#ZusatzDok3E_66ED """ size = 3 command["map_group"] = ord(rom[start_address+1]) command["map_id"] = ord(rom[start_address+2]) elif command_byte == 0xA1: #Hall of Fame code info = "Hall of Fame" long_info = """ #Saves and enters HIRO's complete Team in the Hall of Fame. #Shows the credits and restarts the game with HIRO located in New Bark Town. #[9F] """ size = 1 elif command_byte == 0xA2: #Credits code info = "Credits" long_info = """ #Shows the credits and HIRO is located on the Silver mountain plateau. #[A0] """ size = 1 elif command_byte == 0xA3: #Facing warp info = "Warp-to and set facing direction [Facing (00-03)][Map bank][Map no][X][Y]" long_info = """ #Acts like code 0x3C but defines the desired facing of HIRO. #[A1][Facing (00-03)][Map bank][Map no][X][Y] """ size = 6 command["facing"] = ord(rom[start_address+1]) command["map_group"] = ord(rom[start_address+2]) command["map_id"] = ord(rom[start_address+3]) command["x"] = ord(rom[start_address+4]) command["y"] = ord(rom[start_address+5]) elif command_byte == 0xA4: #MEMORY code [2b + Bank + xx] info = "Set memX to a string by a pointer [aabb][bank][xx]" long_info = """ #MEMORY1, 2 or 3 can directly be filled with a string from #a different rom bank. #[A2][2byte pointer][Bank][00-02 MEMORY] """ size = 5 command["string_pointer"] = calculate_pointer_from_bytes_at(start_address+1, bank="reversed") command["string_pointer_bank"] = ord(rom[start_address+3]) command["memory_id"] = ord(rom[start_address+4]) elif command_byte == 0xA5: #Display any location name [xx] info = "Copy the name of a location (by id) to TEMPMEMORY1" long_info = """ #By the location number the name of that location is written to TEMPMEMORY1. #[A3][Location no] """ size = 2 command["location_number"] = ord(rom[start_address+1]) else: size = 1 #end = True #raise NotImplementedError, "command byte is " + hex(command_byte) + " at " + hex(offset) + " on map " + str(map_group) + "." + str(map_id) print "dunno what this command is: " + hex(command_byte) long_info = clean_up_long_info(long_info) if command_byte in pksv_crystal.keys(): pksv_name = pksv_crystal[command_byte] else: pksv_name = None if command_byte in pksv_no_names.keys(): pksv_no_names[command_byte].append(address) else: pksv_no_names[command_byte] = [address] if debug: print command_debug_information(command_byte=command_byte, map_group=map_group, map_id=map_id, address=offset, info=info, long_info=long_info, pksv_name=pksv_name) #store the size of the command command["size"] = size #the end address is just offset + size - 1 (because size includes command byte) offset += size - 1 #the end address is the last byte belonging to this command command["last_byte_address"] = offset #we also add the size of the command byte to get to the next command offset += 1 #add the command into the command list please commands[len(commands.keys())] = command self.commands = commands script_parse_table[original_start_address : offset] = self return True