mirror of
https://github.com/MarioSpore/Grinch-AP.git
synced 2025-10-21 20:21:32 -06:00
754 lines
34 KiB
Python
754 lines
34 KiB
Python
"""
|
|
Classes and functions related to creating a ROM patch
|
|
"""
|
|
import copy
|
|
import os
|
|
import pkgutil
|
|
from typing import TYPE_CHECKING, Dict, List, Tuple
|
|
|
|
import bsdiff4
|
|
|
|
from worlds.Files import APDeltaPatch
|
|
from settings import get_settings
|
|
|
|
from .data import TrainerPokemonDataTypeEnum, BASE_OFFSET, data
|
|
from .items import reverse_offset_item_value
|
|
from .options import (RandomizeWildPokemon, RandomizeTrainerParties, EliteFourRequirement, NormanRequirement,
|
|
MatchTrainerLevels)
|
|
from .pokemon import HM_MOVES, get_random_move
|
|
from .util import bool_array_to_int, encode_string, get_easter_egg
|
|
|
|
if TYPE_CHECKING:
|
|
from . import PokemonEmeraldWorld
|
|
|
|
|
|
_LOOPING_MUSIC = [
|
|
"MUS_GSC_ROUTE38", "MUS_GSC_PEWTER", "MUS_ROUTE101", "MUS_ROUTE110", "MUS_ROUTE120", "MUS_ROUTE122",
|
|
"MUS_PETALBURG", "MUS_OLDALE", "MUS_GYM", "MUS_SURF", "MUS_PETALBURG_WOODS", "MUS_LILYCOVE_MUSEUM",
|
|
"MUS_OCEANIC_MUSEUM", "MUS_ENCOUNTER_GIRL", "MUS_ENCOUNTER_MALE", "MUS_ABANDONED_SHIP", "MUS_FORTREE",
|
|
"MUS_BIRCH_LAB", "MUS_B_TOWER_RS", "MUS_ENCOUNTER_SWIMMER", "MUS_CAVE_OF_ORIGIN", "MUS_ENCOUNTER_RICH",
|
|
"MUS_VERDANTURF", "MUS_RUSTBORO", "MUS_POKE_CENTER", "MUS_CAUGHT", "MUS_VICTORY_GYM_LEADER", "MUS_VICTORY_LEAGUE",
|
|
"MUS_VICTORY_WILD", "MUS_C_VS_LEGEND_BEAST", "MUS_ROUTE104", "MUS_ROUTE119", "MUS_CYCLING", "MUS_POKE_MART",
|
|
"MUS_LITTLEROOT", "MUS_MT_CHIMNEY", "MUS_ENCOUNTER_FEMALE", "MUS_LILYCOVE", "MUS_DESERT", "MUS_HELP",
|
|
"MUS_UNDERWATER", "MUS_VICTORY_TRAINER", "MUS_ENCOUNTER_MAY", "MUS_ENCOUNTER_INTENSE", "MUS_ENCOUNTER_COOL",
|
|
"MUS_ROUTE113", "MUS_ENCOUNTER_AQUA", "MUS_FOLLOW_ME", "MUS_ENCOUNTER_BRENDAN", "MUS_EVER_GRANDE",
|
|
"MUS_ENCOUNTER_SUSPICIOUS", "MUS_VICTORY_AQUA_MAGMA", "MUS_GAME_CORNER", "MUS_DEWFORD", "MUS_SAFARI_ZONE",
|
|
"MUS_VICTORY_ROAD", "MUS_AQUA_MAGMA_HIDEOUT", "MUS_SAILING", "MUS_MT_PYRE", "MUS_SLATEPORT", "MUS_MT_PYRE_EXTERIOR",
|
|
"MUS_SCHOOL", "MUS_HALL_OF_FAME", "MUS_FALLARBOR", "MUS_SEALED_CHAMBER", "MUS_CONTEST_WINNER", "MUS_CONTEST",
|
|
"MUS_ENCOUNTER_MAGMA", "MUS_ABNORMAL_WEATHER", "MUS_WEATHER_GROUDON", "MUS_SOOTOPOLIS", "MUS_HALL_OF_FAME_ROOM",
|
|
"MUS_TRICK_HOUSE", "MUS_ENCOUNTER_TWINS", "MUS_ENCOUNTER_ELITE_FOUR", "MUS_ENCOUNTER_HIKER", "MUS_CONTEST_LOBBY",
|
|
"MUS_ENCOUNTER_INTERVIEWER", "MUS_ENCOUNTER_CHAMPION", "MUS_B_FRONTIER", "MUS_B_ARENA", "MUS_B_PYRAMID",
|
|
"MUS_B_PYRAMID_TOP", "MUS_B_PALACE", "MUS_B_TOWER", "MUS_B_DOME", "MUS_B_PIKE", "MUS_B_FACTORY", "MUS_VS_RAYQUAZA",
|
|
"MUS_VS_FRONTIER_BRAIN", "MUS_VS_MEW", "MUS_B_DOME_LOBBY", "MUS_VS_WILD", "MUS_VS_AQUA_MAGMA", "MUS_VS_TRAINER",
|
|
"MUS_VS_GYM_LEADER", "MUS_VS_CHAMPION", "MUS_VS_REGI", "MUS_VS_KYOGRE_GROUDON", "MUS_VS_RIVAL", "MUS_VS_ELITE_FOUR",
|
|
"MUS_VS_AQUA_MAGMA_LEADER", "MUS_RG_FOLLOW_ME", "MUS_RG_GAME_CORNER", "MUS_RG_ROCKET_HIDEOUT", "MUS_RG_GYM",
|
|
"MUS_RG_CINNABAR", "MUS_RG_LAVENDER", "MUS_RG_CYCLING", "MUS_RG_ENCOUNTER_ROCKET", "MUS_RG_ENCOUNTER_GIRL",
|
|
"MUS_RG_ENCOUNTER_BOY", "MUS_RG_HALL_OF_FAME", "MUS_RG_VIRIDIAN_FOREST", "MUS_RG_MT_MOON", "MUS_RG_POKE_MANSION",
|
|
"MUS_RG_ROUTE1", "MUS_RG_ROUTE24", "MUS_RG_ROUTE3", "MUS_RG_ROUTE11", "MUS_RG_VICTORY_ROAD", "MUS_RG_VS_GYM_LEADER",
|
|
"MUS_RG_VS_TRAINER", "MUS_RG_VS_WILD", "MUS_RG_VS_CHAMPION", "MUS_RG_PALLET", "MUS_RG_OAK_LAB", "MUS_RG_OAK",
|
|
"MUS_RG_POKE_CENTER", "MUS_RG_SS_ANNE", "MUS_RG_SURF", "MUS_RG_POKE_TOWER", "MUS_RG_SILPH", "MUS_RG_FUCHSIA",
|
|
"MUS_RG_CELADON", "MUS_RG_VICTORY_TRAINER", "MUS_RG_VICTORY_WILD", "MUS_RG_VICTORY_GYM_LEADER", "MUS_RG_VERMILLION",
|
|
"MUS_RG_PEWTER", "MUS_RG_ENCOUNTER_RIVAL", "MUS_RG_RIVAL_EXIT", "MUS_RG_CAUGHT", "MUS_RG_POKE_JUMP",
|
|
"MUS_RG_UNION_ROOM", "MUS_RG_NET_CENTER", "MUS_RG_MYSTERY_GIFT", "MUS_RG_BERRY_PICK", "MUS_RG_SEVII_CAVE",
|
|
"MUS_RG_TEACHY_TV_SHOW", "MUS_RG_SEVII_ROUTE", "MUS_RG_SEVII_DUNGEON", "MUS_RG_SEVII_123", "MUS_RG_SEVII_45",
|
|
"MUS_RG_SEVII_67", "MUS_RG_VS_DEOXYS", "MUS_RG_VS_MEWTWO", "MUS_RG_VS_LEGEND", "MUS_RG_ENCOUNTER_GYM_LEADER",
|
|
"MUS_RG_ENCOUNTER_DEOXYS", "MUS_RG_TRAINER_TOWER", "MUS_RG_SLOW_PALLET", "MUS_RG_TEACHY_TV_MENU",
|
|
]
|
|
|
|
_FANFARES: Dict[str, int] = {
|
|
"MUS_LEVEL_UP": 80,
|
|
"MUS_OBTAIN_ITEM": 160,
|
|
"MUS_EVOLVED": 220,
|
|
"MUS_OBTAIN_TMHM": 220,
|
|
"MUS_HEAL": 160,
|
|
"MUS_OBTAIN_BADGE": 340,
|
|
"MUS_MOVE_DELETED": 180,
|
|
"MUS_OBTAIN_BERRY": 120,
|
|
"MUS_AWAKEN_LEGEND": 710,
|
|
"MUS_SLOTS_JACKPOT": 250,
|
|
"MUS_SLOTS_WIN": 150,
|
|
"MUS_TOO_BAD": 160,
|
|
"MUS_RG_POKE_FLUTE": 450,
|
|
"MUS_RG_OBTAIN_KEY_ITEM": 170,
|
|
"MUS_RG_DEX_RATING": 196,
|
|
"MUS_OBTAIN_B_POINTS": 313,
|
|
"MUS_OBTAIN_SYMBOL": 318,
|
|
"MUS_REGISTER_MATCH_CALL": 135,
|
|
}
|
|
|
|
CAVE_EVENT_NAME_TO_ID = {
|
|
"TERRA_CAVE_ROUTE_114_1": 1,
|
|
"TERRA_CAVE_ROUTE_114_2": 2,
|
|
"TERRA_CAVE_ROUTE_115_1": 3,
|
|
"TERRA_CAVE_ROUTE_115_2": 4,
|
|
"TERRA_CAVE_ROUTE_116_1": 5,
|
|
"TERRA_CAVE_ROUTE_116_2": 6,
|
|
"TERRA_CAVE_ROUTE_118_1": 7,
|
|
"TERRA_CAVE_ROUTE_118_2": 8,
|
|
"MARINE_CAVE_ROUTE_105_1": 9,
|
|
"MARINE_CAVE_ROUTE_105_2": 10,
|
|
"MARINE_CAVE_ROUTE_125_1": 11,
|
|
"MARINE_CAVE_ROUTE_125_2": 12,
|
|
"MARINE_CAVE_ROUTE_127_1": 13,
|
|
"MARINE_CAVE_ROUTE_127_2": 14,
|
|
"MARINE_CAVE_ROUTE_129_1": 15,
|
|
"MARINE_CAVE_ROUTE_129_2": 16,
|
|
}
|
|
|
|
|
|
def _set_bytes_le(byte_array: bytearray, address: int, size: int, value: int) -> None:
|
|
offset = 0
|
|
while size > 0:
|
|
byte_array[address + offset] = value & 0xFF
|
|
value = value >> 8
|
|
offset += 1
|
|
size -= 1
|
|
|
|
|
|
class PokemonEmeraldDeltaPatch(APDeltaPatch):
|
|
game = "Pokemon Emerald"
|
|
hash = "605b89b67018abcea91e693a4dd25be3"
|
|
patch_file_ending = ".apemerald"
|
|
result_file_ending = ".gba"
|
|
|
|
@classmethod
|
|
def get_source_data(cls) -> bytes:
|
|
return get_base_rom_as_bytes()
|
|
|
|
|
|
def create_patch(world: "PokemonEmeraldWorld", output_directory: str) -> None:
|
|
base_rom = get_base_rom_as_bytes()
|
|
base_patch = pkgutil.get_data(__name__, "data/base_patch.bsdiff4")
|
|
patched_rom = bytearray(bsdiff4.patch(base_rom, base_patch))
|
|
|
|
# Set free fly location
|
|
if world.options.free_fly_location:
|
|
_set_bytes_le(
|
|
patched_rom,
|
|
data.rom_addresses["gArchipelagoOptions"] + 0x20,
|
|
1,
|
|
world.free_fly_location_id
|
|
)
|
|
|
|
location_info: List[Tuple[int, int, str]] = []
|
|
for location in world.multiworld.get_locations(world.player):
|
|
if location.address is None:
|
|
continue
|
|
|
|
if location.item is None:
|
|
continue
|
|
|
|
# Set local item values
|
|
if not world.options.remote_items and location.item.player == world.player:
|
|
if type(location.item_address) is int:
|
|
_set_bytes_le(
|
|
patched_rom,
|
|
location.item_address,
|
|
2,
|
|
reverse_offset_item_value(location.item.code)
|
|
)
|
|
elif type(location.item_address) is list:
|
|
for address in location.item_address:
|
|
_set_bytes_le(patched_rom, address, 2, reverse_offset_item_value(location.item.code))
|
|
else:
|
|
if type(location.item_address) is int:
|
|
_set_bytes_le(
|
|
patched_rom,
|
|
location.item_address,
|
|
2,
|
|
data.constants["ITEM_ARCHIPELAGO_PROGRESSION"]
|
|
)
|
|
elif type(location.item_address) is list:
|
|
for address in location.item_address:
|
|
_set_bytes_le(patched_rom, address, 2, data.constants["ITEM_ARCHIPELAGO_PROGRESSION"])
|
|
|
|
# Creates a list of item information to store in tables later. Those tables are used to display the item and
|
|
# player name in a text box. In the case of not enough space, the game will default to "found an ARCHIPELAGO
|
|
# ITEM"
|
|
location_info.append((location.address - BASE_OFFSET, location.item.player, location.item.name))
|
|
|
|
if world.options.trainersanity:
|
|
# Duplicate entries for rival fights
|
|
# For each of the 5 fights, there are 6 variations that have to be accounted for (for starters * genders)
|
|
# The Brendan Mudkip is used as a proxy in the rest of the AP code
|
|
for locale in ["ROUTE_103", "ROUTE_110", "ROUTE_119", "RUSTBORO", "LILYCOVE"]:
|
|
location = world.multiworld.get_location(data.locations[f"TRAINER_BRENDAN_{locale}_MUDKIP_REWARD"].label, world.player)
|
|
alternates = [
|
|
f"TRAINER_BRENDAN_{locale}_TREECKO",
|
|
f"TRAINER_BRENDAN_{locale}_TORCHIC",
|
|
f"TRAINER_MAY_{locale}_MUDKIP",
|
|
f"TRAINER_MAY_{locale}_TREECKO",
|
|
f"TRAINER_MAY_{locale}_TORCHIC",
|
|
]
|
|
location_info.extend((
|
|
data.constants["TRAINER_FLAGS_START"] + data.constants[trainer],
|
|
location.item.player,
|
|
location.item.name
|
|
) for trainer in alternates)
|
|
|
|
player_name_ids: Dict[str, int] = {world.multiworld.player_name[world.player]: 0}
|
|
item_name_offsets: Dict[str, int] = {}
|
|
next_item_name_offset = 0
|
|
for i, (flag, item_player, item_name) in enumerate(sorted(location_info, key=lambda t: t[0])):
|
|
# The player's own items are still set in the table with the value 0 to indicate the game should not show any
|
|
# message (the message for receiving an item will pop up when the client eventually gives it to them).
|
|
# In race mode, no item location data is included, and only recieved (or own) items will show any text box.
|
|
if item_player == world.player or world.multiworld.is_race:
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gArchipelagoNameTable"] + (i * 5) + 0, 2, flag)
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gArchipelagoNameTable"] + (i * 5) + 2, 2, 0)
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gArchipelagoNameTable"] + (i * 5) + 4, 1, 0)
|
|
else:
|
|
player_name = world.multiworld.player_name[item_player]
|
|
|
|
if player_name not in player_name_ids:
|
|
# Only space for 50 player names
|
|
if len(player_name_ids) >= 50:
|
|
continue
|
|
|
|
player_name_ids[player_name] = len(player_name_ids)
|
|
for j, b in enumerate(encode_string(player_name, 17)):
|
|
_set_bytes_le(
|
|
patched_rom,
|
|
data.rom_addresses["gArchipelagoPlayerNames"] + (player_name_ids[player_name] * 17) + j,
|
|
1,
|
|
b
|
|
)
|
|
|
|
if item_name not in item_name_offsets:
|
|
if len(item_name) > 35:
|
|
item_name = item_name[:34] + "…"
|
|
|
|
# Only 36 * 250 bytes for item names
|
|
if next_item_name_offset + len(item_name) + 1 > 36 * 250:
|
|
continue
|
|
|
|
item_name_offsets[item_name] = next_item_name_offset
|
|
next_item_name_offset += len(item_name) + 1
|
|
for j, b in enumerate(encode_string(item_name) + b"\xFF"):
|
|
_set_bytes_le(
|
|
patched_rom,
|
|
data.rom_addresses["gArchipelagoItemNames"] + (item_name_offsets[item_name]) + j,
|
|
1,
|
|
b
|
|
)
|
|
|
|
# There should always be enough space for one entry per location
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gArchipelagoNameTable"] + (i * 5) + 0, 2, flag)
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gArchipelagoNameTable"] + (i * 5) + 2, 2, item_name_offsets[item_name])
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gArchipelagoNameTable"] + (i * 5) + 4, 1, player_name_ids[player_name])
|
|
|
|
easter_egg = get_easter_egg(world.options.easter_egg.value)
|
|
|
|
# Set start inventory
|
|
start_inventory = world.options.start_inventory.value.copy()
|
|
|
|
starting_badges = 0
|
|
if start_inventory.pop("Stone Badge", 0) > 0:
|
|
starting_badges |= (1 << 0)
|
|
if start_inventory.pop("Knuckle Badge", 0) > 0:
|
|
starting_badges |= (1 << 1)
|
|
if start_inventory.pop("Dynamo Badge", 0) > 0:
|
|
starting_badges |= (1 << 2)
|
|
if start_inventory.pop("Heat Badge", 0) > 0:
|
|
starting_badges |= (1 << 3)
|
|
if start_inventory.pop("Balance Badge", 0) > 0:
|
|
starting_badges |= (1 << 4)
|
|
if start_inventory.pop("Feather Badge", 0) > 0:
|
|
starting_badges |= (1 << 5)
|
|
if start_inventory.pop("Mind Badge", 0) > 0:
|
|
starting_badges |= (1 << 6)
|
|
if start_inventory.pop("Rain Badge", 0) > 0:
|
|
starting_badges |= (1 << 7)
|
|
|
|
pc_slots: List[Tuple[str, int]] = []
|
|
while any(qty > 0 for qty in start_inventory.values()):
|
|
if len(pc_slots) >= 19:
|
|
break
|
|
|
|
for i, item_name in enumerate(start_inventory.keys()):
|
|
if len(pc_slots) >= 19:
|
|
break
|
|
|
|
quantity = min(start_inventory[item_name], 999)
|
|
if quantity == 0:
|
|
continue
|
|
|
|
start_inventory[item_name] -= quantity
|
|
|
|
pc_slots.append((item_name, quantity))
|
|
|
|
pc_slots.sort(reverse=True)
|
|
|
|
for i, slot in enumerate(pc_slots):
|
|
address = data.rom_addresses["sNewGamePCItems"] + (i * 4)
|
|
item = reverse_offset_item_value(world.item_name_to_id[slot[0]])
|
|
_set_bytes_le(patched_rom, address + 0, 2, item)
|
|
_set_bytes_le(patched_rom, address + 2, 2, slot[1])
|
|
|
|
# Set species data
|
|
_set_species_info(world, patched_rom, easter_egg)
|
|
|
|
# Set encounter tables
|
|
if world.options.wild_pokemon != RandomizeWildPokemon.option_vanilla:
|
|
_set_encounter_tables(world, patched_rom)
|
|
|
|
# Set opponent data
|
|
if world.options.trainer_parties != RandomizeTrainerParties.option_vanilla or easter_egg[0] == 2:
|
|
_set_opponents(world, patched_rom, easter_egg)
|
|
|
|
# Set legendary pokemon
|
|
_set_legendary_encounters(world, patched_rom)
|
|
|
|
# Set misc pokemon
|
|
_set_misc_pokemon(world, patched_rom)
|
|
|
|
# Set starters
|
|
_set_starters(world, patched_rom)
|
|
|
|
# Set TM moves
|
|
_set_tm_moves(world, patched_rom, easter_egg)
|
|
|
|
# Randomize move tutor moves
|
|
_randomize_move_tutor_moves(world, patched_rom, easter_egg)
|
|
|
|
# Set TM/HM compatibility
|
|
_set_tmhm_compatibility(world, patched_rom)
|
|
|
|
# Randomize opponent double or single
|
|
_randomize_opponent_battle_type(world, patched_rom)
|
|
|
|
# Options
|
|
# struct ArchipelagoOptions
|
|
# {
|
|
# /* 0x00 */ u16 birchPokemon;
|
|
# /* 0x02 */ bool8 advanceTextWithHoldA;
|
|
# /* 0x03 */ u8 receivedItemMessageFilter; // 0 = Show All; 1 = Show Progression Only; 2 = Show None
|
|
# /* 0x04 */ bool8 betterShopsEnabled;
|
|
# /* 0x05 */ bool8 reusableTms;
|
|
# /* 0x06 */ bool8 guaranteedCatch;
|
|
# /* 0x07 */ bool8 purgeSpinners;
|
|
# /* 0x08 */ bool8 areTrainersBlind;
|
|
# /* 0x09 */ u16 expMultiplierNumerator;
|
|
# /* 0x0B */ u16 expMultiplierDenominator;
|
|
# /* 0x0D */ bool8 matchTrainerLevels;
|
|
# /* 0x0E */ s8 matchTrainerLevelBonus;
|
|
# /* 0x0F */ bool8 eliteFourRequiresGyms;
|
|
# /* 0x10 */ u8 eliteFourRequiredCount;
|
|
# /* 0x11 */ bool8 normanRequiresGyms;
|
|
# /* 0x12 */ u8 normanRequiredCount;
|
|
# /* 0x13 */ u8 startingBadges;
|
|
# /* 0x14 */ u32 hmTotalBadgeRequirements;
|
|
# /* 0x18 */ u8 hmSpecificBadgeRequirements[8];
|
|
# /* 0x20 */ u8 freeFlyLocation;
|
|
# /* 0x21 */ u8 terraCaveLocationId:4;
|
|
# u8 marineCaveLocationId:4;
|
|
# /* 0x22 */ bool8 addRoute115Boulders;
|
|
# /* 0x23 */ bool8 addBumpySlopes;
|
|
# /* 0x24 */ bool8 modifyRoute118;
|
|
# /* 0x25 */ u16 removedBlockers;
|
|
# /* 0x27 */ bool8 berryTreesRandomized;
|
|
# /* 0x28 */ bool8 isDexsanity;
|
|
# /* 0x29 */ bool8 isTrainersanity;
|
|
# /* 0x2A */ bool8 isWarpRando;
|
|
# /* 0x2B */ u8 activeEasterEgg;
|
|
# /* 0x2C */ bool8 normalizeEncounterRates;
|
|
# /* 0x2D */ bool8 allowWonderTrading;
|
|
# /* 0x2E */ u16 matchTrainerLevelMultiplierNumerator;
|
|
# /* 0x30 */ u16 matchTrainerLevelMultiplierDenominator;
|
|
# /* 0x32 */ bool8 allowSkippingFanfares;
|
|
# };
|
|
options_address = data.rom_addresses["gArchipelagoOptions"]
|
|
|
|
# Set Birch pokemon
|
|
_set_bytes_le(
|
|
patched_rom,
|
|
options_address + 0x00,
|
|
2,
|
|
world.random.choice(list(data.species.keys()))
|
|
)
|
|
|
|
# Set hold A to advance text
|
|
_set_bytes_le(patched_rom, options_address + 0x02, 1, 1 if world.options.turbo_a else 0)
|
|
|
|
# Set receive item messages type
|
|
_set_bytes_le(patched_rom, options_address + 0x03, 1, world.options.receive_item_messages.value)
|
|
|
|
# Set better shops
|
|
_set_bytes_le(patched_rom, options_address + 0x04, 1, 1 if world.options.better_shops else 0)
|
|
|
|
# Set reusable TMs
|
|
_set_bytes_le(patched_rom, options_address + 0x05, 1, 1 if world.options.reusable_tms_tutors else 0)
|
|
|
|
# Set guaranteed catch
|
|
_set_bytes_le(patched_rom, options_address + 0x06, 1, 1 if world.options.guaranteed_catch else 0)
|
|
|
|
# Set purge spinners
|
|
_set_bytes_le(patched_rom, options_address + 0x07, 1, 1 if world.options.purge_spinners else 0)
|
|
|
|
# Set blind trainers
|
|
_set_bytes_le(patched_rom, options_address + 0x08, 1, 1 if world.options.blind_trainers else 0)
|
|
|
|
# Set exp modifier
|
|
_set_bytes_le(patched_rom, options_address + 0x09, 2, min(max(world.options.exp_modifier.value, 0), 2**16 - 1))
|
|
_set_bytes_le(patched_rom, options_address + 0x0B, 2, 100)
|
|
|
|
# Set match trainer levels
|
|
_set_bytes_le(patched_rom, options_address + 0x0D, 1, 1 if world.options.match_trainer_levels else 0)
|
|
|
|
# Set match trainer levels bonus
|
|
if world.options.match_trainer_levels == MatchTrainerLevels.option_additive:
|
|
match_trainer_levels_bonus = max(min(world.options.match_trainer_levels_bonus.value, 100), -100)
|
|
_set_bytes_le(patched_rom, options_address + 0x0E, 1, match_trainer_levels_bonus) # Works with negatives
|
|
elif world.options.match_trainer_levels == MatchTrainerLevels.option_multiplicative:
|
|
_set_bytes_le(patched_rom, options_address + 0x2E, 2, world.options.match_trainer_levels_bonus.value + 100)
|
|
_set_bytes_le(patched_rom, options_address + 0x30, 2, 100)
|
|
|
|
# Set elite four requirement
|
|
_set_bytes_le(
|
|
patched_rom,
|
|
options_address + 0x0F,
|
|
1,
|
|
1 if world.options.elite_four_requirement == EliteFourRequirement.option_gyms else 0
|
|
)
|
|
|
|
# Set elite four count
|
|
_set_bytes_le(patched_rom, options_address + 0x10, 1, min(max(world.options.elite_four_count.value, 0), 8))
|
|
|
|
# Set norman requirement
|
|
_set_bytes_le(
|
|
patched_rom,
|
|
options_address + 0x11,
|
|
1,
|
|
1 if world.options.norman_requirement == NormanRequirement.option_gyms else 0
|
|
)
|
|
|
|
# Set norman count
|
|
_set_bytes_le(patched_rom, options_address + 0x12, 1, min(max(world.options.norman_count.value, 0), 8))
|
|
|
|
# Set starting badges
|
|
_set_bytes_le(patched_rom, options_address + 0x13, 1, starting_badges)
|
|
|
|
# Set HM badge requirements
|
|
field_move_order = [
|
|
"HM01 Cut",
|
|
"HM05 Flash",
|
|
"HM06 Rock Smash",
|
|
"HM04 Strength",
|
|
"HM03 Surf",
|
|
"HM02 Fly",
|
|
"HM08 Dive",
|
|
"HM07 Waterfall",
|
|
]
|
|
badge_to_bit = {
|
|
"Stone Badge": 1 << 0,
|
|
"Knuckle Badge": 1 << 1,
|
|
"Dynamo Badge": 1 << 2,
|
|
"Heat Badge": 1 << 3,
|
|
"Balance Badge": 1 << 4,
|
|
"Feather Badge": 1 << 5,
|
|
"Mind Badge": 1 << 6,
|
|
"Rain Badge": 1 << 7,
|
|
}
|
|
|
|
# Number of badges
|
|
# Uses 4 bits per HM. 0-8 means it's a valid requirement, otherwise use specific badges.
|
|
hm_badge_counts = 0
|
|
for i, hm in enumerate(field_move_order):
|
|
hm_badge_counts |= (world.hm_requirements[hm] if isinstance(world.hm_requirements[hm], int) else 0xF) << (i * 4)
|
|
_set_bytes_le(patched_rom, options_address + 0x14, 4, hm_badge_counts)
|
|
|
|
# Specific badges
|
|
for i, hm in enumerate(field_move_order):
|
|
if isinstance(world.hm_requirements, list):
|
|
bitfield = 0
|
|
for badge in world.hm_requirements:
|
|
bitfield |= badge_to_bit[badge]
|
|
_set_bytes_le(patched_rom, options_address + 0x18 + i, 1, bitfield)
|
|
|
|
# Set terra/marine cave locations
|
|
terra_cave_id = CAVE_EVENT_NAME_TO_ID[world.multiworld.get_location("TERRA_CAVE_LOCATION", world.player).item.name]
|
|
marine_cave_id = CAVE_EVENT_NAME_TO_ID[world.multiworld.get_location("MARINE_CAVE_LOCATION", world.player).item.name]
|
|
_set_bytes_le(patched_rom, options_address + 0x21, 1, terra_cave_id | (marine_cave_id << 4))
|
|
|
|
# Set route 115 boulders
|
|
_set_bytes_le(patched_rom, options_address + 0x22, 1, 1 if world.options.extra_boulders else 0)
|
|
|
|
# Swap route 115 layout if bumpy slope enabled
|
|
_set_bytes_le(patched_rom, options_address + 0x23, 1, 1 if world.options.extra_bumpy_slope else 0)
|
|
|
|
# Swap route 115 layout if bumpy slope enabled
|
|
_set_bytes_le(patched_rom, options_address + 0x24, 1, 1 if world.options.modify_118 else 0)
|
|
|
|
# Set removed blockers
|
|
removed_roadblocks = world.options.remove_roadblocks.value
|
|
removed_roadblocks_bitfield = 0
|
|
removed_roadblocks_bitfield |= (1 << 0) if "Safari Zone Construction Workers" in removed_roadblocks else 0
|
|
removed_roadblocks_bitfield |= (1 << 1) if "Lilycove City Wailmer" in removed_roadblocks else 0
|
|
removed_roadblocks_bitfield |= (1 << 2) if "Route 110 Aqua Grunts" in removed_roadblocks else 0
|
|
removed_roadblocks_bitfield |= (1 << 3) if "Aqua Hideout Grunts" in removed_roadblocks else 0
|
|
removed_roadblocks_bitfield |= (1 << 4) if "Route 119 Aqua Grunts" in removed_roadblocks else 0
|
|
removed_roadblocks_bitfield |= (1 << 5) if "Route 112 Magma Grunts" in removed_roadblocks else 0
|
|
removed_roadblocks_bitfield |= (1 << 6) if "Seafloor Cavern Aqua Grunt" in removed_roadblocks else 0
|
|
_set_bytes_le(patched_rom, options_address + 0x25, 2, removed_roadblocks_bitfield)
|
|
|
|
# Mark berry trees as randomized
|
|
_set_bytes_le(patched_rom, options_address + 0x27, 1, 1 if world.options.berry_trees else 0)
|
|
|
|
# Mark dexsanity as enabled
|
|
_set_bytes_le(patched_rom, options_address + 0x28, 1, 1 if world.options.dexsanity else 0)
|
|
|
|
# Mark trainersanity as enabled
|
|
_set_bytes_le(patched_rom, options_address + 0x29, 1, 1 if world.options.trainersanity else 0)
|
|
|
|
# Set easter egg data
|
|
_set_bytes_le(patched_rom, options_address + 0x2B, 1, easter_egg[0])
|
|
|
|
# Set normalize encounter rates
|
|
_set_bytes_le(patched_rom, options_address + 0x2C, 1, 1 if world.options.normalize_encounter_rates else 0)
|
|
|
|
# Set allow wonder trading
|
|
_set_bytes_le(patched_rom, options_address + 0x2D, 1, 1 if world.options.enable_wonder_trading else 0)
|
|
|
|
# Set allowed to skip fanfares
|
|
_set_bytes_le(patched_rom, options_address + 0x32, 1, 1 if world.options.fanfares else 0)
|
|
|
|
if easter_egg[0] == 2:
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gBattleMoves"] + (easter_egg[1] * 12) + 4, 1, 50)
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gBattleMoves"] + (data.constants["MOVE_CUT"] * 12) + 4, 1, 1)
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gBattleMoves"] + (data.constants["MOVE_FLY"] * 12) + 4, 1, 1)
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gBattleMoves"] + (data.constants["MOVE_SURF"] * 12) + 4, 1, 1)
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gBattleMoves"] + (data.constants["MOVE_STRENGTH"] * 12) + 4, 1, 1)
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gBattleMoves"] + (data.constants["MOVE_FLASH"] * 12) + 4, 1, 1)
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gBattleMoves"] + (data.constants["MOVE_ROCK_SMASH"] * 12) + 4, 1, 1)
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gBattleMoves"] + (data.constants["MOVE_WATERFALL"] * 12) + 4, 1, 1)
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gBattleMoves"] + (data.constants["MOVE_DIVE"] * 12) + 4, 1, 1)
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gBattleMoves"] + (data.constants["MOVE_DIG"] * 12) + 4, 1, 1)
|
|
|
|
# Set slot auth
|
|
for i, byte in enumerate(world.auth):
|
|
_set_bytes_le(patched_rom, data.rom_addresses["gArchipelagoInfo"] + i, 1, byte)
|
|
|
|
# Randomize music
|
|
if world.options.music:
|
|
# The "randomized sound table" is a patchboard that redirects sounds just before they get played
|
|
randomized_looping_music = copy.copy(_LOOPING_MUSIC)
|
|
world.random.shuffle(randomized_looping_music)
|
|
for original_music, randomized_music in zip(_LOOPING_MUSIC, randomized_looping_music):
|
|
_set_bytes_le(
|
|
patched_rom,
|
|
data.rom_addresses["gRandomizedSoundTable"] + (data.constants[original_music] * 2),
|
|
2,
|
|
data.constants[randomized_music]
|
|
)
|
|
|
|
# Randomize fanfares
|
|
if world.options.fanfares:
|
|
# Shuffle the lists, pair new tracks with original tracks, set the new track ids, and set new fanfare durations
|
|
randomized_fanfares = [fanfare_name for fanfare_name in _FANFARES]
|
|
world.random.shuffle(randomized_fanfares)
|
|
for i, fanfare_pair in enumerate(zip(_FANFARES.keys(), randomized_fanfares)):
|
|
_set_bytes_le(
|
|
patched_rom,
|
|
data.rom_addresses["gRandomizedSoundTable"] + (data.constants[fanfare_pair[0]] * 2),
|
|
2,
|
|
data.constants[fanfare_pair[1]]
|
|
)
|
|
_set_bytes_le(
|
|
patched_rom,
|
|
data.rom_addresses["sFanfares"] + (i * 4) + 2,
|
|
2,
|
|
_FANFARES[fanfare_pair[1]]
|
|
)
|
|
|
|
# Write Output
|
|
out_file_name = world.multiworld.get_out_file_name_base(world.player)
|
|
output_path = os.path.join(output_directory, f"{out_file_name}.gba")
|
|
with open(output_path, "wb") as out_file:
|
|
out_file.write(patched_rom)
|
|
patch = PokemonEmeraldDeltaPatch(os.path.splitext(output_path)[0] + ".apemerald", player=world.player,
|
|
player_name=world.multiworld.get_player_name(world.player),
|
|
patched_path=output_path)
|
|
|
|
patch.write()
|
|
os.unlink(output_path)
|
|
|
|
|
|
def get_base_rom_as_bytes() -> bytes:
|
|
with open(get_settings().pokemon_emerald_settings.rom_file, "rb") as infile:
|
|
base_rom_bytes = bytes(infile.read())
|
|
|
|
return base_rom_bytes
|
|
|
|
|
|
def _set_encounter_tables(world: "PokemonEmeraldWorld", rom: bytearray) -> None:
|
|
"""
|
|
Encounter tables are lists of
|
|
struct {
|
|
min_level: 0x01 bytes,
|
|
max_level: 0x01 bytes,
|
|
species_id: 0x02 bytes
|
|
}
|
|
"""
|
|
for map_data in world.modified_maps.values():
|
|
tables = [map_data.land_encounters, map_data.water_encounters, map_data.fishing_encounters]
|
|
for table in tables:
|
|
if table is not None:
|
|
for i, species_id in enumerate(table.slots):
|
|
address = table.address + 2 + (4 * i)
|
|
_set_bytes_le(rom, address, 2, species_id)
|
|
|
|
|
|
def _set_species_info(world: "PokemonEmeraldWorld", rom: bytearray, easter_egg: Tuple[int, int]) -> None:
|
|
for species in world.modified_species.values():
|
|
_set_bytes_le(rom, species.address + 6, 1, species.types[0])
|
|
_set_bytes_le(rom, species.address + 7, 1, species.types[1])
|
|
_set_bytes_le(rom, species.address + 8, 1, species.catch_rate)
|
|
_set_bytes_le(rom, species.address + 22, 1, species.abilities[0])
|
|
_set_bytes_le(rom, species.address + 23, 1, species.abilities[1])
|
|
|
|
if easter_egg[0] == 3:
|
|
_set_bytes_le(rom, species.address + 22, 1, easter_egg[1])
|
|
_set_bytes_le(rom, species.address + 23, 1, easter_egg[1])
|
|
|
|
for i, learnset_move in enumerate(species.learnset):
|
|
level_move = learnset_move.level << 9 | learnset_move.move_id
|
|
if easter_egg[0] == 2:
|
|
level_move = learnset_move.level << 9 | easter_egg[1]
|
|
|
|
_set_bytes_le(rom, species.learnset_address + (i * 2), 2, level_move)
|
|
|
|
|
|
def _set_opponents(world: "PokemonEmeraldWorld", rom: bytearray, easter_egg: Tuple[int, int]) -> None:
|
|
for trainer in world.modified_trainers:
|
|
party_address = trainer.party.address
|
|
|
|
pokemon_data_size: int
|
|
if trainer.party.pokemon_data_type in {TrainerPokemonDataTypeEnum.NO_ITEM_DEFAULT_MOVES, TrainerPokemonDataTypeEnum.ITEM_DEFAULT_MOVES}:
|
|
pokemon_data_size = 8
|
|
else: # Custom Moves
|
|
pokemon_data_size = 16
|
|
|
|
for i, pokemon in enumerate(trainer.party.pokemon):
|
|
pokemon_address = party_address + (i * pokemon_data_size)
|
|
|
|
# Replace species
|
|
_set_bytes_le(rom, pokemon_address + 0x04, 2, pokemon.species_id)
|
|
|
|
# Replace custom moves if applicable
|
|
if trainer.party.pokemon_data_type == TrainerPokemonDataTypeEnum.NO_ITEM_CUSTOM_MOVES:
|
|
if easter_egg[0] == 2:
|
|
_set_bytes_le(rom, pokemon_address + 0x06, 2, easter_egg[1])
|
|
_set_bytes_le(rom, pokemon_address + 0x08, 2, easter_egg[1])
|
|
_set_bytes_le(rom, pokemon_address + 0x0A, 2, easter_egg[1])
|
|
_set_bytes_le(rom, pokemon_address + 0x0C, 2, easter_egg[1])
|
|
else:
|
|
_set_bytes_le(rom, pokemon_address + 0x06, 2, pokemon.moves[0])
|
|
_set_bytes_le(rom, pokemon_address + 0x08, 2, pokemon.moves[1])
|
|
_set_bytes_le(rom, pokemon_address + 0x0A, 2, pokemon.moves[2])
|
|
_set_bytes_le(rom, pokemon_address + 0x0C, 2, pokemon.moves[3])
|
|
elif trainer.party.pokemon_data_type == TrainerPokemonDataTypeEnum.ITEM_CUSTOM_MOVES:
|
|
if easter_egg[0] == 2:
|
|
_set_bytes_le(rom, pokemon_address + 0x08, 2, easter_egg[1])
|
|
_set_bytes_le(rom, pokemon_address + 0x0A, 2, easter_egg[1])
|
|
_set_bytes_le(rom, pokemon_address + 0x0C, 2, easter_egg[1])
|
|
_set_bytes_le(rom, pokemon_address + 0x0E, 2, easter_egg[1])
|
|
else:
|
|
_set_bytes_le(rom, pokemon_address + 0x08, 2, pokemon.moves[0])
|
|
_set_bytes_le(rom, pokemon_address + 0x0A, 2, pokemon.moves[1])
|
|
_set_bytes_le(rom, pokemon_address + 0x0C, 2, pokemon.moves[2])
|
|
_set_bytes_le(rom, pokemon_address + 0x0E, 2, pokemon.moves[3])
|
|
|
|
|
|
def _set_legendary_encounters(world: "PokemonEmeraldWorld", rom: bytearray) -> None:
|
|
for encounter in world.modified_legendary_encounters:
|
|
_set_bytes_le(rom, encounter.address, 2, encounter.species_id)
|
|
|
|
|
|
def _set_misc_pokemon(world: "PokemonEmeraldWorld", rom: bytearray) -> None:
|
|
for encounter in world.modified_misc_pokemon:
|
|
_set_bytes_le(rom, encounter.address, 2, encounter.species_id)
|
|
|
|
|
|
def _set_starters(world: "PokemonEmeraldWorld", rom: bytearray) -> None:
|
|
address = data.rom_addresses["sStarterMon"]
|
|
(starter_1, starter_2, starter_3) = world.modified_starters
|
|
|
|
_set_bytes_le(rom, address + 0, 2, starter_1)
|
|
_set_bytes_le(rom, address + 2, 2, starter_2)
|
|
_set_bytes_le(rom, address + 4, 2, starter_3)
|
|
|
|
|
|
def _set_tm_moves(world: "PokemonEmeraldWorld", rom: bytearray, easter_egg: Tuple[int, int]) -> None:
|
|
tmhm_list_address = data.rom_addresses["sTMHMMoves"]
|
|
|
|
for i, move in enumerate(world.modified_tmhm_moves):
|
|
# Don't modify HMs
|
|
if i >= 50:
|
|
break
|
|
|
|
_set_bytes_le(rom, tmhm_list_address + (i * 2), 2, move)
|
|
if easter_egg[0] == 2:
|
|
_set_bytes_le(rom, tmhm_list_address + (i * 2), 2, easter_egg[1])
|
|
|
|
|
|
def _set_tmhm_compatibility(world: "PokemonEmeraldWorld", rom: bytearray) -> None:
|
|
learnsets_address = data.rom_addresses["gTMHMLearnsets"]
|
|
|
|
for species in world.modified_species.values():
|
|
_set_bytes_le(rom, learnsets_address + (species.species_id * 8), 8, species.tm_hm_compatibility)
|
|
|
|
|
|
def _randomize_opponent_battle_type(world: "PokemonEmeraldWorld", rom: bytearray) -> None:
|
|
probability = world.options.double_battle_chance.value / 100
|
|
|
|
battle_type_map = {
|
|
0: 4,
|
|
1: 8,
|
|
2: 6,
|
|
3: 13,
|
|
}
|
|
|
|
for trainer_data in data.trainers:
|
|
if trainer_data.script_address != 0 and len(trainer_data.party.pokemon) > 1:
|
|
original_battle_type = rom[trainer_data.script_address + 1]
|
|
if original_battle_type in battle_type_map: # Don't touch anything other than regular single battles
|
|
if world.random.random() < probability:
|
|
# Set the trainer to be a double battle
|
|
_set_bytes_le(rom, trainer_data.address + 0x18, 1, 1)
|
|
|
|
# Swap the battle type in the script for the purpose of loading the right text
|
|
# and setting data to the right places
|
|
_set_bytes_le(
|
|
rom,
|
|
trainer_data.script_address + 1,
|
|
1,
|
|
battle_type_map[original_battle_type]
|
|
)
|
|
|
|
|
|
def _randomize_move_tutor_moves(world: "PokemonEmeraldWorld", rom: bytearray, easter_egg: Tuple[int, int]) -> None:
|
|
if easter_egg[0] == 2:
|
|
for i in range(30):
|
|
_set_bytes_le(rom, data.rom_addresses["gTutorMoves"] + (i * 2), 2, easter_egg[1])
|
|
else:
|
|
if world.options.tm_tutor_moves:
|
|
new_tutor_moves = []
|
|
for i in range(30):
|
|
new_move = get_random_move(world.random, set(new_tutor_moves) | world.blacklisted_moves | HM_MOVES)
|
|
new_tutor_moves.append(new_move)
|
|
|
|
_set_bytes_le(rom, data.rom_addresses["gTutorMoves"] + (i * 2), 2, new_move)
|
|
|
|
# Always set Fortree move tutor to Dig
|
|
_set_bytes_le(rom, data.rom_addresses["gTutorMoves"] + (24 * 2), 2, data.constants["MOVE_DIG"])
|
|
|
|
# Modify compatibility
|
|
if world.options.tm_tutor_compatibility.value != -1:
|
|
for species in data.species.values():
|
|
_set_bytes_le(
|
|
rom,
|
|
data.rom_addresses["sTutorLearnsets"] + (species.species_id * 4),
|
|
4,
|
|
bool_array_to_int([world.random.randrange(0, 100) < world.options.tm_tutor_compatibility.value for _ in range(32)])
|
|
)
|