2021-07-22 15:51:50 +02:00
|
|
|
import random
|
2021-08-09 06:50:11 +02:00
|
|
|
import logging
|
2021-08-09 09:15:41 +02:00
|
|
|
import os
|
|
|
|
import threading
|
2021-08-28 00:26:02 +02:00
|
|
|
import typing
|
2021-07-22 15:51:50 +02:00
|
|
|
|
2022-05-11 13:05:53 -05:00
|
|
|
from BaseClasses import Item, CollectionState, Tutorial
|
2021-07-12 15:11:48 +02:00
|
|
|
from .SubClasses import ALttPItem
|
2022-05-11 13:05:53 -05:00
|
|
|
from ..AutoWorld import World, WebWorld, LogicMixin
|
2021-08-30 09:59:20 -07:00
|
|
|
from .Options import alttp_options, smallkey_shuffle
|
2022-05-19 09:37:26 -04:00
|
|
|
from .Items import as_dict_item_table, item_name_groups, item_table, GetBeemizerItem
|
2021-07-22 15:51:50 +02:00
|
|
|
from .Regions import lookup_name_to_id, create_regions, mark_light_world_regions
|
|
|
|
from .Rules import set_rules
|
2021-08-28 00:26:02 +02:00
|
|
|
from .ItemPool import generate_itempool, difficulties
|
2021-08-30 01:18:30 +02:00
|
|
|
from .Shops import create_shops, ShopSlotFill
|
2021-07-22 15:51:50 +02:00
|
|
|
from .Dungeons import create_dungeons
|
2021-09-18 22:13:19 +02:00
|
|
|
from .Rom import LocalRom, patch_rom, patch_race_rom, patch_enemizer, apply_rom_settings, get_hash_string, \
|
2022-03-18 04:53:09 +01:00
|
|
|
get_base_rom_path, LttPDeltaPatch
|
2021-08-09 09:15:41 +02:00
|
|
|
import Patch
|
2022-05-19 09:37:26 -04:00
|
|
|
from itertools import chain
|
2021-07-12 13:54:47 +02:00
|
|
|
|
2021-07-22 15:51:50 +02:00
|
|
|
from .InvertedRegions import create_inverted_regions, mark_dark_world_regions
|
|
|
|
from .EntranceShuffle import link_entrances, link_inverted_entrances, plando_connect
|
2020-10-24 05:38:56 +02:00
|
|
|
|
2021-08-09 06:50:11 +02:00
|
|
|
lttp_logger = logging.getLogger("A Link to the Past")
|
|
|
|
|
2022-06-10 07:23:03 -04:00
|
|
|
extras_list = sum(difficulties['normal'].extras[0:5], [])
|
2021-08-28 00:26:02 +02:00
|
|
|
|
2022-05-11 13:05:53 -05:00
|
|
|
class ALTTPWeb(WebWorld):
|
|
|
|
setup_en = Tutorial(
|
|
|
|
"Multiworld Setup Tutorial",
|
|
|
|
"A guide to setting up the Archipelago ALttP Software on your computer. This guide covers single-player, multiworld, and related software.",
|
|
|
|
"English",
|
|
|
|
"multiworld_en.md",
|
|
|
|
"multiworld/en",
|
|
|
|
["Farrak Kilhn"]
|
|
|
|
)
|
|
|
|
|
|
|
|
setup_de = Tutorial(
|
|
|
|
setup_en.tutorial_name,
|
|
|
|
setup_en.description,
|
|
|
|
"Deutsch",
|
|
|
|
"multiworld_de.md",
|
|
|
|
"multiworld/de",
|
|
|
|
["Fischfilet"]
|
|
|
|
)
|
|
|
|
|
|
|
|
setup_es = Tutorial(
|
|
|
|
setup_en.tutorial_name,
|
|
|
|
setup_en.description,
|
|
|
|
"Español",
|
|
|
|
"multiworld_es.md",
|
|
|
|
"multiworld/es",
|
|
|
|
["Edos"]
|
|
|
|
)
|
|
|
|
|
|
|
|
setup_fr = Tutorial(
|
|
|
|
setup_en.tutorial_name,
|
|
|
|
setup_en.description,
|
|
|
|
"Français",
|
|
|
|
"multiworld_fr.md",
|
|
|
|
"multiworld/fr",
|
|
|
|
["Coxla"]
|
|
|
|
)
|
|
|
|
|
|
|
|
msu = Tutorial(
|
|
|
|
"MSU-1 Setup Tutorial",
|
|
|
|
"A guide to setting up MSU-1, which allows for custom in-game music.",
|
|
|
|
"English",
|
|
|
|
"msu1_en.md",
|
|
|
|
"msu1/en",
|
|
|
|
["Farrak Kilhn"]
|
|
|
|
)
|
|
|
|
|
|
|
|
msu_es = Tutorial(
|
|
|
|
msu.tutorial_name,
|
|
|
|
msu.description,
|
|
|
|
"Español",
|
|
|
|
"msu1_es.md",
|
2022-06-25 06:15:57 -05:00
|
|
|
"msu1/es",
|
2022-05-11 13:05:53 -05:00
|
|
|
["Edos"]
|
|
|
|
)
|
|
|
|
|
|
|
|
msu_fr = Tutorial(
|
|
|
|
msu.tutorial_name,
|
|
|
|
msu.description,
|
|
|
|
"Français",
|
|
|
|
"msu1_fr.md",
|
|
|
|
"msu1/fr",
|
|
|
|
["Coxla"]
|
|
|
|
)
|
|
|
|
|
|
|
|
plando = Tutorial(
|
|
|
|
"Plando Tutorial",
|
|
|
|
"A guide to creating Multiworld Plandos with LTTP",
|
|
|
|
"English",
|
|
|
|
"plando_en.md",
|
|
|
|
"plando/en",
|
|
|
|
["Berserker"]
|
|
|
|
)
|
|
|
|
|
|
|
|
tutorials = [setup_en, setup_de, setup_es, setup_fr, msu, msu_es, msu_fr, plando]
|
|
|
|
|
|
|
|
|
2021-06-11 14:22:44 +02:00
|
|
|
class ALTTPWorld(World):
|
2021-08-31 17:28:46 -04:00
|
|
|
"""
|
|
|
|
The Legend of Zelda: A Link to the Past is an action/adventure game. Take on the role of
|
|
|
|
Link, a boy who is destined to save the land of Hyrule. Delve through three palaces and nine
|
|
|
|
dungeons on your quest to rescue the descendents of the seven wise men and defeat the evil
|
|
|
|
Ganon!
|
|
|
|
"""
|
2021-06-11 14:22:44 +02:00
|
|
|
game: str = "A Link to the Past"
|
2021-07-04 16:18:21 +02:00
|
|
|
options = alttp_options
|
2021-07-08 11:07:41 +02:00
|
|
|
topology_present = True
|
2021-07-12 13:54:47 +02:00
|
|
|
item_name_groups = item_name_groups
|
2021-07-12 15:33:20 +02:00
|
|
|
hint_blacklist = {"Triforce"}
|
2021-07-08 11:07:41 +02:00
|
|
|
|
2021-07-12 18:05:46 +02:00
|
|
|
item_name_to_id = {name: data.item_code for name, data in item_table.items() if type(data.item_code) == int}
|
|
|
|
location_name_to_id = lookup_name_to_id
|
|
|
|
|
2021-08-27 14:52:33 +02:00
|
|
|
data_version = 8
|
2021-07-13 19:14:57 +02:00
|
|
|
remote_items: bool = False
|
2021-09-23 03:48:37 +02:00
|
|
|
remote_start_inventory: bool = False
|
2022-04-08 11:16:36 +02:00
|
|
|
required_client_version = (0, 3, 2)
|
2022-05-11 13:05:53 -05:00
|
|
|
web = ALTTPWeb()
|
2021-07-13 19:14:57 +02:00
|
|
|
|
2021-07-22 15:51:50 +02:00
|
|
|
set_rules = set_rules
|
|
|
|
|
|
|
|
create_items = generate_itempool
|
|
|
|
|
2021-08-30 16:31:56 +02:00
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
self.dungeon_local_item_names = set()
|
|
|
|
self.dungeon_specific_item_names = set()
|
|
|
|
self.rom_name_available_event = threading.Event()
|
2021-09-20 01:00:09 +02:00
|
|
|
self.has_progressive_bows = False
|
2021-08-30 16:31:56 +02:00
|
|
|
super(ALTTPWorld, self).__init__(*args, **kwargs)
|
|
|
|
|
2022-04-29 20:37:28 -05:00
|
|
|
@classmethod
|
|
|
|
def stage_assert_generate(cls, world):
|
|
|
|
rom_file = get_base_rom_path()
|
|
|
|
if not os.path.exists(rom_file):
|
|
|
|
raise FileNotFoundError(rom_file)
|
|
|
|
|
2021-08-28 00:26:02 +02:00
|
|
|
def generate_early(self):
|
|
|
|
player = self.player
|
|
|
|
world = self.world
|
2021-08-30 16:31:56 +02:00
|
|
|
|
2021-08-28 00:26:02 +02:00
|
|
|
# system for sharing ER layouts
|
2021-10-06 11:32:49 +02:00
|
|
|
self.er_seed = str(world.random.randint(0, 2 ** 64))
|
2021-08-28 00:26:02 +02:00
|
|
|
|
|
|
|
if "-" in world.shuffle[player]:
|
|
|
|
shuffle, seed = world.shuffle[player].split("-", 1)
|
|
|
|
world.shuffle[player] = shuffle
|
|
|
|
if shuffle == "vanilla":
|
2021-10-06 11:32:49 +02:00
|
|
|
self.er_seed = "vanilla"
|
2021-08-28 00:26:02 +02:00
|
|
|
elif seed.startswith("group-") or world.is_race:
|
2021-10-06 11:32:49 +02:00
|
|
|
self.er_seed = get_same_seed(world, (
|
2022-06-01 17:29:21 +02:00
|
|
|
shuffle, seed, world.retro_caves[player], world.mode[player], world.logic[player]))
|
2021-08-28 00:26:02 +02:00
|
|
|
else: # not a race or group seed, use set seed as is.
|
2021-10-06 11:32:49 +02:00
|
|
|
self.er_seed = seed
|
2021-08-28 00:26:02 +02:00
|
|
|
elif world.shuffle[player] == "vanilla":
|
2021-10-06 11:32:49 +02:00
|
|
|
self.er_seed = "vanilla"
|
2021-08-30 18:00:39 +02:00
|
|
|
for dungeon_item in ["smallkey_shuffle", "bigkey_shuffle", "compass_shuffle", "map_shuffle"]:
|
2021-08-30 16:31:56 +02:00
|
|
|
option = getattr(world, dungeon_item)[player]
|
|
|
|
if option == "own_world":
|
2021-09-17 00:17:54 +02:00
|
|
|
world.local_items[player].value |= self.item_name_groups[option.item_name_group]
|
2021-08-30 16:31:56 +02:00
|
|
|
elif option == "different_world":
|
2021-09-17 00:17:54 +02:00
|
|
|
world.non_local_items[player].value |= self.item_name_groups[option.item_name_group]
|
2021-08-30 16:31:56 +02:00
|
|
|
elif option.in_dungeon:
|
|
|
|
self.dungeon_local_item_names |= self.item_name_groups[option.item_name_group]
|
|
|
|
if option == "original_dungeon":
|
|
|
|
self.dungeon_specific_item_names |= self.item_name_groups[option.item_name_group]
|
2021-08-28 00:26:02 +02:00
|
|
|
|
|
|
|
world.difficulty_requirements[player] = difficulties[world.difficulty[player]]
|
|
|
|
|
2021-07-22 15:51:50 +02:00
|
|
|
def create_regions(self):
|
2021-07-23 12:03:19 +02:00
|
|
|
player = self.player
|
2021-07-22 15:51:50 +02:00
|
|
|
world = self.world
|
2021-07-23 12:03:19 +02:00
|
|
|
if world.open_pyramid[player] == 'goal':
|
|
|
|
world.open_pyramid[player] = world.goal[player] in {'crystals', 'ganontriforcehunt',
|
|
|
|
'localganontriforcehunt', 'ganonpedestal'}
|
|
|
|
elif world.open_pyramid[player] == 'auto':
|
|
|
|
world.open_pyramid[player] = world.goal[player] in {'crystals', 'ganontriforcehunt',
|
|
|
|
'localganontriforcehunt', 'ganonpedestal'} and \
|
|
|
|
(world.shuffle[player] in {'vanilla', 'dungeonssimple', 'dungeonsfull',
|
|
|
|
'dungeonscrossed'} or not world.shuffle_ganon)
|
|
|
|
else:
|
|
|
|
world.open_pyramid[player] = {'on': True, 'off': False, 'yes': True, 'no': False}.get(
|
|
|
|
world.open_pyramid[player], 'auto')
|
|
|
|
|
|
|
|
world.triforce_pieces_available[player] = max(world.triforce_pieces_available[player],
|
|
|
|
world.triforce_pieces_required[player])
|
|
|
|
|
|
|
|
if world.mode[player] != 'inverted':
|
|
|
|
create_regions(world, player)
|
|
|
|
else:
|
|
|
|
create_inverted_regions(world, player)
|
|
|
|
create_shops(world, player)
|
|
|
|
create_dungeons(world, player)
|
|
|
|
|
|
|
|
if world.logic[player] not in ["noglitches", "minorglitches"] and world.shuffle[player] in \
|
|
|
|
{"vanilla", "dungeonssimple", "dungeonsfull", "simple", "restricted", "full"}:
|
|
|
|
world.fix_fake_world[player] = False
|
|
|
|
|
|
|
|
# seeded entrance shuffle
|
|
|
|
old_random = world.random
|
2021-10-06 11:32:49 +02:00
|
|
|
world.random = random.Random(self.er_seed)
|
2021-07-23 12:03:19 +02:00
|
|
|
|
|
|
|
if world.mode[player] != 'inverted':
|
|
|
|
link_entrances(world, player)
|
|
|
|
mark_light_world_regions(world, player)
|
|
|
|
else:
|
|
|
|
link_inverted_entrances(world, player)
|
|
|
|
mark_dark_world_regions(world, player)
|
|
|
|
|
|
|
|
world.random = old_random
|
|
|
|
plando_connect(world, player)
|
2021-07-22 15:51:50 +02:00
|
|
|
|
2021-08-21 06:55:08 +02:00
|
|
|
def collect_item(self, state: CollectionState, item: Item, remove=False):
|
|
|
|
item_name = item.name
|
|
|
|
if item_name.startswith('Progressive '):
|
|
|
|
if remove:
|
|
|
|
if 'Sword' in item_name:
|
|
|
|
if state.has('Golden Sword', item.player):
|
|
|
|
return 'Golden Sword'
|
|
|
|
elif state.has('Tempered Sword', item.player):
|
|
|
|
return 'Tempered Sword'
|
|
|
|
elif state.has('Master Sword', item.player):
|
|
|
|
return 'Master Sword'
|
|
|
|
elif state.has('Fighter Sword', item.player):
|
|
|
|
return 'Fighter Sword'
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
elif 'Glove' in item.name:
|
|
|
|
if state.has('Titans Mitts', item.player):
|
|
|
|
return 'Titans Mitts'
|
|
|
|
elif state.has('Power Glove', item.player):
|
|
|
|
return 'Power Glove'
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
elif 'Shield' in item_name:
|
|
|
|
if state.has('Mirror Shield', item.player):
|
|
|
|
return 'Mirror Shield'
|
|
|
|
elif state.has('Red Shield', item.player):
|
|
|
|
return 'Red Shield'
|
|
|
|
elif state.has('Blue Shield', item.player):
|
|
|
|
return 'Blue Shield'
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
elif 'Bow' in item_name:
|
|
|
|
if state.has('Silver Bow', item.player):
|
|
|
|
return 'Silver Bow'
|
|
|
|
elif state.has('Bow', item.player):
|
|
|
|
return 'Bow'
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
if 'Sword' in item_name:
|
|
|
|
if state.has('Golden Sword', item.player):
|
|
|
|
pass
|
|
|
|
elif state.has('Tempered Sword', item.player) and self.world.difficulty_requirements[
|
|
|
|
item.player].progressive_sword_limit >= 4:
|
|
|
|
return 'Golden Sword'
|
|
|
|
elif state.has('Master Sword', item.player) and self.world.difficulty_requirements[
|
|
|
|
item.player].progressive_sword_limit >= 3:
|
|
|
|
return 'Tempered Sword'
|
|
|
|
elif state.has('Fighter Sword', item.player) and self.world.difficulty_requirements[item.player].progressive_sword_limit >= 2:
|
|
|
|
return 'Master Sword'
|
|
|
|
elif self.world.difficulty_requirements[item.player].progressive_sword_limit >= 1:
|
|
|
|
return 'Fighter Sword'
|
|
|
|
elif 'Glove' in item_name:
|
|
|
|
if state.has('Titans Mitts', item.player):
|
|
|
|
return
|
|
|
|
elif state.has('Power Glove', item.player):
|
|
|
|
return 'Titans Mitts'
|
|
|
|
else:
|
|
|
|
return 'Power Glove'
|
|
|
|
elif 'Shield' in item_name:
|
|
|
|
if state.has('Mirror Shield', item.player):
|
|
|
|
return
|
|
|
|
elif state.has('Red Shield', item.player) and self.world.difficulty_requirements[item.player].progressive_shield_limit >= 3:
|
|
|
|
return 'Mirror Shield'
|
2022-02-17 06:07:11 +01:00
|
|
|
elif state.has('Blue Shield', item.player) and self.world.difficulty_requirements[item.player].progressive_shield_limit >= 2:
|
2021-08-21 06:55:08 +02:00
|
|
|
return 'Red Shield'
|
|
|
|
elif self.world.difficulty_requirements[item.player].progressive_shield_limit >= 1:
|
|
|
|
return 'Blue Shield'
|
|
|
|
elif 'Bow' in item_name:
|
2021-08-26 16:03:22 -05:00
|
|
|
if state.has('Silver Bow', item.player):
|
2021-08-21 06:55:08 +02:00
|
|
|
return
|
2021-08-31 17:28:46 -04:00
|
|
|
elif state.has('Bow', item.player) and (self.world.difficulty_requirements[item.player].progressive_bow_limit >= 2
|
|
|
|
or self.world.logic[item.player] == 'noglitches'
|
2021-08-26 16:03:22 -05:00
|
|
|
or self.world.swordless[item.player]): # modes where silver bow is always required for ganon
|
2021-08-21 06:55:08 +02:00
|
|
|
return 'Silver Bow'
|
|
|
|
elif self.world.difficulty_requirements[item.player].progressive_bow_limit >= 1:
|
|
|
|
return 'Bow'
|
2021-08-10 09:47:28 +02:00
|
|
|
elif item.advancement:
|
2021-08-21 06:55:08 +02:00
|
|
|
return item_name
|
2020-10-24 05:38:56 +02:00
|
|
|
|
2021-08-09 06:50:11 +02:00
|
|
|
def pre_fill(self):
|
|
|
|
from Fill import fill_restrictive, FillError
|
|
|
|
attempts = 5
|
|
|
|
world = self.world
|
|
|
|
player = self.player
|
2021-09-01 14:01:54 -05:00
|
|
|
all_state = world.get_all_state(use_cache=True)
|
2021-08-09 06:50:11 +02:00
|
|
|
crystals = [self.create_item(name) for name in ['Red Pendant', 'Blue Pendant', 'Green Pendant', 'Crystal 1', 'Crystal 2', 'Crystal 3', 'Crystal 4', 'Crystal 7', 'Crystal 5', 'Crystal 6']]
|
|
|
|
crystal_locations = [world.get_location('Turtle Rock - Prize', player),
|
|
|
|
world.get_location('Eastern Palace - Prize', player),
|
|
|
|
world.get_location('Desert Palace - Prize', player),
|
|
|
|
world.get_location('Tower of Hera - Prize', player),
|
|
|
|
world.get_location('Palace of Darkness - Prize', player),
|
|
|
|
world.get_location('Thieves\' Town - Prize', player),
|
|
|
|
world.get_location('Skull Woods - Prize', player),
|
|
|
|
world.get_location('Swamp Palace - Prize', player),
|
|
|
|
world.get_location('Ice Palace - Prize', player),
|
|
|
|
world.get_location('Misery Mire - Prize', player)]
|
|
|
|
placed_prizes = {loc.item.name for loc in crystal_locations if loc.item}
|
|
|
|
unplaced_prizes = [crystal for crystal in crystals if crystal.name not in placed_prizes]
|
|
|
|
empty_crystal_locations = [loc for loc in crystal_locations if not loc.item]
|
|
|
|
for attempt in range(attempts):
|
|
|
|
try:
|
|
|
|
prizepool = unplaced_prizes.copy()
|
|
|
|
prize_locs = empty_crystal_locations.copy()
|
|
|
|
world.random.shuffle(prize_locs)
|
|
|
|
fill_restrictive(world, all_state, prize_locs, prizepool, True, lock=True)
|
|
|
|
except FillError as e:
|
|
|
|
lttp_logger.exception("Failed to place dungeon prizes (%s). Will retry %s more times", e,
|
|
|
|
attempts - attempt)
|
|
|
|
for location in empty_crystal_locations:
|
|
|
|
location.item = None
|
|
|
|
continue
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
raise FillError('Unable to place dungeon prizes')
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def stage_pre_fill(cls, world):
|
|
|
|
from .Dungeons import fill_dungeons_restrictive
|
2022-02-13 23:02:18 +01:00
|
|
|
fill_dungeons_restrictive(world)
|
2021-08-09 06:50:11 +02:00
|
|
|
|
2021-08-30 01:16:04 +02:00
|
|
|
@classmethod
|
|
|
|
def stage_post_fill(cls, world):
|
2021-08-30 01:18:30 +02:00
|
|
|
ShopSlotFill(world)
|
2021-08-30 01:16:04 +02:00
|
|
|
|
2021-08-09 09:15:41 +02:00
|
|
|
def generate_output(self, output_directory: str):
|
|
|
|
world = self.world
|
|
|
|
player = self.player
|
2021-08-10 09:03:44 +02:00
|
|
|
try:
|
|
|
|
use_enemizer = (world.boss_shuffle[player] != 'none' or world.enemy_shuffle[player]
|
|
|
|
or world.enemy_health[player] != 'default' or world.enemy_damage[player] != 'default'
|
2021-09-13 01:32:32 +02:00
|
|
|
or world.pot_shuffle[player] or world.bush_shuffle[player]
|
2021-08-10 09:03:44 +02:00
|
|
|
or world.killable_thieves[player])
|
|
|
|
|
2021-09-18 22:13:19 +02:00
|
|
|
rom = LocalRom(get_base_rom_path())
|
2021-08-10 09:03:44 +02:00
|
|
|
|
|
|
|
patch_rom(world, rom, player, use_enemizer)
|
|
|
|
|
|
|
|
if use_enemizer:
|
|
|
|
patch_enemizer(world, player, rom, world.enemizer, output_directory)
|
|
|
|
|
|
|
|
if world.is_race:
|
|
|
|
patch_race_rom(rom, world, player)
|
|
|
|
|
|
|
|
world.spoiler.hashes[player] = get_hash_string(rom.hash)
|
|
|
|
|
|
|
|
palettes_options = {
|
|
|
|
'dungeon': world.uw_palettes[player],
|
|
|
|
'overworld': world.ow_palettes[player],
|
|
|
|
'hud': world.hud_palettes[player],
|
|
|
|
'sword': world.sword_palettes[player],
|
|
|
|
'shield': world.shield_palettes[player],
|
|
|
|
'link': world.link_palettes[player]
|
|
|
|
}
|
|
|
|
palettes_options = {key: option.current_key for key, option in palettes_options.items()}
|
|
|
|
|
|
|
|
apply_rom_settings(rom, world.heartbeep[player].current_key,
|
|
|
|
world.heartcolor[player].current_key,
|
|
|
|
world.quickswap[player],
|
|
|
|
world.menuspeed[player].current_key,
|
|
|
|
world.music[player],
|
|
|
|
world.sprite[player],
|
|
|
|
palettes_options, world, player, True,
|
|
|
|
reduceflashing=world.reduceflashing[player] or world.is_race,
|
2021-11-08 16:34:54 +01:00
|
|
|
triforcehud=world.triforcehud[player].current_key,
|
2022-04-04 18:54:49 -07:00
|
|
|
deathlink=world.death_link[player],
|
|
|
|
allowcollect=world.allow_collect[player])
|
2021-08-10 09:03:44 +02:00
|
|
|
|
|
|
|
outfilepname = f'_P{player}'
|
2022-04-02 22:47:42 -05:00
|
|
|
outfilepname += f"_{world.get_file_safe_player_name(player).replace(' ', '_')}" \
|
2021-08-10 09:03:44 +02:00
|
|
|
if world.player_name[player] != 'Player%d' % player else ''
|
|
|
|
|
|
|
|
rompath = os.path.join(output_directory, f'AP_{world.seed_name}{outfilepname}.sfc')
|
2021-09-15 01:02:06 +02:00
|
|
|
rom.write_to_file(rompath)
|
2022-03-18 04:53:09 +01:00
|
|
|
patch = LttPDeltaPatch(os.path.splitext(rompath)[0]+LttPDeltaPatch.patch_file_ending, player=player,
|
|
|
|
player_name=world.player_name[player], patched_path=rompath)
|
|
|
|
patch.write()
|
2021-08-10 09:03:44 +02:00
|
|
|
os.unlink(rompath)
|
|
|
|
self.rom_name = rom.name
|
|
|
|
except:
|
|
|
|
raise
|
|
|
|
finally:
|
|
|
|
self.rom_name_available_event.set() # make sure threading continues and errors are collected
|
2021-08-09 09:15:41 +02:00
|
|
|
|
|
|
|
def modify_multidata(self, multidata: dict):
|
|
|
|
import base64
|
|
|
|
# wait for self.rom_name to be available.
|
|
|
|
self.rom_name_available_event.wait()
|
2021-08-10 09:03:44 +02:00
|
|
|
rom_name = getattr(self, "rom_name", None)
|
|
|
|
# we skip in case of error, so that the original error in the output thread is the one that gets raised
|
|
|
|
if rom_name:
|
|
|
|
new_name = base64.b64encode(bytes(self.rom_name)).decode()
|
2022-02-09 21:06:34 +01:00
|
|
|
multidata["connect_names"][new_name] = multidata["connect_names"][self.world.player_name[self.player]]
|
2021-08-09 09:15:41 +02:00
|
|
|
|
2021-07-12 13:54:47 +02:00
|
|
|
def create_item(self, name: str) -> Item:
|
|
|
|
return ALttPItem(name, self.player, **as_dict_item_table[name])
|
|
|
|
|
2021-08-10 09:03:44 +02:00
|
|
|
@classmethod
|
2021-08-30 22:20:44 +02:00
|
|
|
def stage_fill_hook(cls, world, progitempool, nonexcludeditempool, localrestitempool, nonlocalrestitempool,
|
|
|
|
restitempool, fill_locations):
|
2021-08-10 09:03:44 +02:00
|
|
|
trash_counts = {}
|
|
|
|
standard_keyshuffle_players = set()
|
|
|
|
for player in world.get_game_players("A Link to the Past"):
|
2021-08-30 18:00:39 +02:00
|
|
|
if world.mode[player] == 'standard' and world.smallkey_shuffle[player] \
|
2022-01-09 04:48:31 +01:00
|
|
|
and world.smallkey_shuffle[player] != smallkey_shuffle.option_universal and \
|
|
|
|
world.smallkey_shuffle[player] != smallkey_shuffle.option_own_dungeons:
|
2021-08-10 09:03:44 +02:00
|
|
|
standard_keyshuffle_players.add(player)
|
|
|
|
if not world.ganonstower_vanilla[player] or \
|
|
|
|
world.logic[player] in {'owglitches', 'hybridglitches', "nologic"}:
|
|
|
|
pass
|
|
|
|
elif 'triforcehunt' in world.goal[player] and ('local' in world.goal[player] or world.players == 1):
|
|
|
|
trash_counts[player] = world.random.randint(world.crystals_needed_for_gt[player] * 2,
|
|
|
|
world.crystals_needed_for_gt[player] * 4)
|
|
|
|
else:
|
|
|
|
trash_counts[player] = world.random.randint(0, world.crystals_needed_for_gt[player] * 2)
|
|
|
|
|
|
|
|
# Make sure the escape small key is placed first in standard with key shuffle to prevent running out of spots
|
2021-08-14 00:37:58 +02:00
|
|
|
# TODO: this might be worthwhile to introduce as generic option for various games and then optimize it
|
2021-08-10 09:03:44 +02:00
|
|
|
if standard_keyshuffle_players:
|
2022-01-09 04:32:17 +01:00
|
|
|
viable = {}
|
2021-08-14 00:37:58 +02:00
|
|
|
for location in world.get_locations():
|
|
|
|
if location.player in standard_keyshuffle_players \
|
|
|
|
and location.item is None \
|
|
|
|
and location.can_reach(world.state):
|
2022-01-09 04:32:17 +01:00
|
|
|
viable.setdefault(location.player, []).append(location)
|
|
|
|
|
2021-08-14 00:37:58 +02:00
|
|
|
for player in standard_keyshuffle_players:
|
2022-01-09 04:32:17 +01:00
|
|
|
loc = world.random.choice(viable[player])
|
2021-08-14 00:37:58 +02:00
|
|
|
key = world.create_item("Small Key (Hyrule Castle)", player)
|
|
|
|
loc.place_locked_item(key)
|
|
|
|
fill_locations.remove(loc)
|
|
|
|
world.random.shuffle(fill_locations)
|
|
|
|
# TODO: investigate not creating the key in the first place
|
2022-02-13 23:02:18 +01:00
|
|
|
progitempool[:] = [item for item in progitempool if
|
|
|
|
item.player not in standard_keyshuffle_players or
|
|
|
|
item.name != "Small Key (Hyrule Castle)"]
|
2021-08-10 09:03:44 +02:00
|
|
|
|
|
|
|
if trash_counts:
|
|
|
|
locations_mapping = {player: [] for player in trash_counts}
|
|
|
|
for location in fill_locations:
|
|
|
|
if 'Ganons Tower' in location.name and location.player in locations_mapping:
|
|
|
|
locations_mapping[location.player].append(location)
|
|
|
|
|
|
|
|
for player, trash_count in trash_counts.items():
|
|
|
|
gtower_locations = locations_mapping[player]
|
|
|
|
world.random.shuffle(gtower_locations)
|
|
|
|
localrest = localrestitempool[player]
|
|
|
|
if localrest:
|
|
|
|
gt_item_pool = restitempool + localrest
|
|
|
|
world.random.shuffle(gt_item_pool)
|
|
|
|
else:
|
|
|
|
gt_item_pool = restitempool.copy()
|
|
|
|
|
|
|
|
while gtower_locations and gt_item_pool and trash_count > 0:
|
|
|
|
spot_to_fill = gtower_locations.pop()
|
|
|
|
item_to_place = gt_item_pool.pop()
|
2022-06-17 03:24:15 +02:00
|
|
|
if spot_to_fill.item_rule(item_to_place):
|
|
|
|
if item_to_place in localrest:
|
|
|
|
localrest.remove(item_to_place)
|
|
|
|
else:
|
|
|
|
restitempool.remove(item_to_place)
|
|
|
|
world.push_item(spot_to_fill, item_to_place, False)
|
|
|
|
fill_locations.remove(spot_to_fill) # very slow, unfortunately
|
|
|
|
trash_count -= 1
|
2021-08-28 00:26:02 +02:00
|
|
|
|
2022-06-10 07:23:03 -04:00
|
|
|
|
2022-02-05 15:49:19 +01:00
|
|
|
def get_filler_item_name(self) -> str:
|
2022-05-19 09:37:26 -04:00
|
|
|
if self.world.goal[self.player] == "icerodhunt":
|
|
|
|
item = "Nothing"
|
|
|
|
else:
|
2022-06-10 07:23:03 -04:00
|
|
|
item = self.world.random.choice(extras_list)
|
2022-05-19 09:37:26 -04:00
|
|
|
return GetBeemizerItem(self.world, self.player, item)
|
2022-02-05 15:49:19 +01:00
|
|
|
|
2022-02-13 23:02:18 +01:00
|
|
|
def get_pre_fill_items(self):
|
|
|
|
res = []
|
|
|
|
if self.dungeon_local_item_names:
|
|
|
|
for (name, player), dungeon in self.world.dungeons.items():
|
|
|
|
if player == self.player:
|
|
|
|
for item in dungeon.all_items:
|
|
|
|
if item.name in self.dungeon_local_item_names:
|
|
|
|
res.append(item)
|
|
|
|
return res
|
|
|
|
|
2021-08-28 00:26:02 +02:00
|
|
|
|
|
|
|
def get_same_seed(world, seed_def: tuple) -> str:
|
|
|
|
seeds: typing.Dict[tuple, str] = getattr(world, "__named_seeds", {})
|
|
|
|
if seed_def in seeds:
|
|
|
|
return seeds[seed_def]
|
|
|
|
seeds[seed_def] = str(world.random.randint(0, 2 ** 64))
|
|
|
|
world.__named_seeds = seeds
|
2021-08-30 19:11:12 +02:00
|
|
|
return seeds[seed_def]
|
|
|
|
|
|
|
|
|
|
|
|
class ALttPLogic(LogicMixin):
|
|
|
|
def _lttp_has_key(self, item, player, count: int = 1):
|
|
|
|
if self.world.logic[player] == 'nologic':
|
|
|
|
return True
|
|
|
|
if self.world.smallkey_shuffle[player] == smallkey_shuffle.option_universal:
|
|
|
|
return self.can_buy_unlimited('Small Key (Universal)', player)
|
2021-08-31 17:28:46 -04:00
|
|
|
return self.prog_items[item, player] >= count
|