2024-02-28 04:44:22 +01:00
|
|
|
import logging
|
2024-08-20 01:34:40 +02:00
|
|
|
import math
|
2024-02-28 04:44:22 +01:00
|
|
|
from dataclasses import dataclass
|
2024-08-20 01:34:40 +02:00
|
|
|
from typing import TYPE_CHECKING, Any, Dict, List, Optional, Set, Tuple, Union, cast
|
2024-04-12 00:27:42 +02:00
|
|
|
|
2024-08-20 01:34:40 +02:00
|
|
|
from BaseClasses import CollectionState, Item, Location, LocationProgressType, MultiWorld, Region
|
2024-04-12 00:27:42 +02:00
|
|
|
|
|
|
|
from .data import static_logic as static_witness_logic
|
|
|
|
from .data.utils import weighted_sample
|
2024-07-02 23:59:26 +02:00
|
|
|
from .player_items import WitnessItem
|
2023-11-24 06:27:03 +01:00
|
|
|
|
|
|
|
if TYPE_CHECKING:
|
|
|
|
from . import WitnessWorld
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-08-20 01:34:40 +02:00
|
|
|
CompactHintArgs = Tuple[Union[str, int], Union[str, int]]
|
|
|
|
CompactHintData = Tuple[str, Union[str, int], Union[str, int]]
|
2024-03-05 22:54:02 +01:00
|
|
|
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
@dataclass
|
|
|
|
class WitnessLocationHint:
|
|
|
|
location: Location
|
|
|
|
hint_came_from_location: bool
|
|
|
|
|
|
|
|
# If a hint gets added to a set twice, but once as an item hint and once as a location hint, those are the same
|
2024-04-12 00:27:42 +02:00
|
|
|
def __hash__(self) -> int:
|
2024-02-28 04:44:22 +01:00
|
|
|
return hash(self.location)
|
|
|
|
|
2024-07-02 23:59:26 +02:00
|
|
|
def __eq__(self, other: Any) -> bool:
|
|
|
|
if not isinstance(other, WitnessLocationHint):
|
|
|
|
return False
|
2024-02-28 04:44:22 +01:00
|
|
|
return self.location == other.location
|
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
class WitnessWordedHint:
|
|
|
|
wording: str
|
|
|
|
location: Optional[Location] = None
|
|
|
|
area: Optional[str] = None
|
|
|
|
area_amount: Optional[int] = None
|
2024-08-20 01:16:35 +02:00
|
|
|
area_hunt_panels: Optional[int] = None
|
2024-08-20 01:34:40 +02:00
|
|
|
vague_location_hint: bool = False
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
|
2023-12-10 20:36:55 +01:00
|
|
|
def get_always_hint_items(world: "WitnessWorld") -> List[str]:
|
2023-06-21 00:45:26 +02:00
|
|
|
always = [
|
2022-10-09 04:13:52 +02:00
|
|
|
"Boat",
|
2023-11-24 06:27:03 +01:00
|
|
|
"Caves Shortcuts",
|
2023-02-01 21:18:07 +01:00
|
|
|
"Progressive Dots",
|
2022-10-09 04:13:52 +02:00
|
|
|
]
|
|
|
|
|
2023-11-24 06:27:03 +01:00
|
|
|
difficulty = world.options.puzzle_randomization
|
|
|
|
discards = world.options.shuffle_discarded_panels
|
|
|
|
wincon = world.options.victory_condition
|
2022-10-09 04:13:52 +02:00
|
|
|
|
|
|
|
if discards:
|
2024-02-11 02:25:03 +01:00
|
|
|
if difficulty == "sigma_expert":
|
2023-06-21 00:45:26 +02:00
|
|
|
always.append("Arrows")
|
2022-10-09 04:13:52 +02:00
|
|
|
else:
|
2023-06-21 00:45:26 +02:00
|
|
|
always.append("Triangles")
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-02-11 02:25:03 +01:00
|
|
|
if wincon == "elevator":
|
2024-02-28 04:44:22 +01:00
|
|
|
always += ["Mountain Bottom Floor Pillars Room Entry (Door)", "Mountain Bottom Floor Doors"]
|
2023-11-24 06:27:03 +01:00
|
|
|
|
2024-02-11 02:25:03 +01:00
|
|
|
if wincon == "challenge":
|
2023-11-24 06:27:03 +01:00
|
|
|
always += ["Challenge Entry (Panel)", "Caves Panels"]
|
2023-06-21 00:45:26 +02:00
|
|
|
|
|
|
|
return always
|
2022-10-09 04:13:52 +02:00
|
|
|
|
|
|
|
|
2024-01-16 13:14:38 +01:00
|
|
|
def get_always_hint_locations(world: "WitnessWorld") -> List[str]:
|
|
|
|
always = [
|
2022-10-09 04:13:52 +02:00
|
|
|
"Challenge Vault Box",
|
|
|
|
"Mountain Bottom Floor Discard",
|
2023-02-01 21:18:07 +01:00
|
|
|
"Theater Eclipse EP",
|
|
|
|
"Shipwreck Couch EP",
|
|
|
|
"Mountainside Cloud Cycle EP",
|
2023-12-10 20:36:55 +01:00
|
|
|
]
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-01-16 13:14:38 +01:00
|
|
|
# Add Obelisk Sides that contain EPs that are meant to be hinted, if they are necessary to complete the Obelisk Side
|
2024-02-28 04:44:22 +01:00
|
|
|
if "0x339B6" not in world.player_logic.COMPLETELY_DISABLED_ENTITIES:
|
2024-01-16 13:14:38 +01:00
|
|
|
always.append("Town Obelisk Side 6") # Eclipse EP
|
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
if "0x3388F" not in world.player_logic.COMPLETELY_DISABLED_ENTITIES:
|
2024-01-16 13:14:38 +01:00
|
|
|
always.append("Treehouse Obelisk Side 4") # Couch EP
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
if "0x335AE" not in world.player_logic.COMPLETELY_DISABLED_ENTITIES:
|
|
|
|
always.append("Mountainside Obelisk Side 1") # Cloud Cycle EP.
|
2024-01-16 13:14:38 +01:00
|
|
|
|
|
|
|
return always
|
|
|
|
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2023-12-10 20:36:55 +01:00
|
|
|
def get_priority_hint_items(world: "WitnessWorld") -> List[str]:
|
2022-10-09 04:13:52 +02:00
|
|
|
priority = {
|
2023-06-21 00:45:26 +02:00
|
|
|
"Caves Mountain Shortcut (Door)",
|
|
|
|
"Caves Swamp Shortcut (Door)",
|
2022-10-09 04:13:52 +02:00
|
|
|
"Swamp Entry (Panel)",
|
|
|
|
"Swamp Laser Shortcut (Door)",
|
|
|
|
}
|
|
|
|
|
2023-11-24 06:27:03 +01:00
|
|
|
if world.options.shuffle_symbols:
|
|
|
|
symbols = [
|
|
|
|
"Progressive Dots",
|
|
|
|
"Progressive Stars",
|
|
|
|
"Shapers",
|
|
|
|
"Rotated Shapers",
|
|
|
|
"Negative Shapers",
|
|
|
|
"Arrows",
|
|
|
|
"Triangles",
|
|
|
|
"Eraser",
|
|
|
|
"Black/White Squares",
|
|
|
|
"Colored Squares",
|
|
|
|
"Sound Dots",
|
2024-01-16 15:13:04 +01:00
|
|
|
"Progressive Symmetry"
|
2023-11-24 06:27:03 +01:00
|
|
|
]
|
|
|
|
|
|
|
|
priority.update(world.random.sample(symbols, 5))
|
|
|
|
|
|
|
|
if world.options.shuffle_lasers:
|
2023-07-18 10:18:42 +02:00
|
|
|
lasers = [
|
2022-10-09 04:13:52 +02:00
|
|
|
"Symmetry Laser",
|
|
|
|
"Town Laser",
|
|
|
|
"Keep Laser",
|
|
|
|
"Swamp Laser",
|
|
|
|
"Treehouse Laser",
|
|
|
|
"Monastery Laser",
|
|
|
|
"Jungle Laser",
|
|
|
|
"Quarry Laser",
|
|
|
|
"Bunker Laser",
|
|
|
|
"Shadows Laser",
|
2023-07-18 10:18:42 +02:00
|
|
|
]
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2023-11-24 06:27:03 +01:00
|
|
|
if world.options.shuffle_doors >= 2:
|
2022-10-09 04:13:52 +02:00
|
|
|
priority.add("Desert Laser")
|
2023-11-24 06:27:03 +01:00
|
|
|
priority.update(world.random.sample(lasers, 5))
|
2022-10-09 04:13:52 +02:00
|
|
|
|
|
|
|
else:
|
2023-07-18 10:18:42 +02:00
|
|
|
lasers.append("Desert Laser")
|
2023-11-24 06:27:03 +01:00
|
|
|
priority.update(world.random.sample(lasers, 6))
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2023-12-10 20:36:55 +01:00
|
|
|
return sorted(priority)
|
2022-10-09 04:13:52 +02:00
|
|
|
|
|
|
|
|
2024-01-16 13:14:38 +01:00
|
|
|
def get_priority_hint_locations(world: "WitnessWorld") -> List[str]:
|
|
|
|
priority = [
|
2024-02-28 04:44:22 +01:00
|
|
|
"Tutorial Patio Floor",
|
|
|
|
"Tutorial Patio Flowers EP",
|
2023-06-21 00:45:26 +02:00
|
|
|
"Swamp Purple Underwater",
|
|
|
|
"Shipwreck Vault Box",
|
2024-02-28 04:44:22 +01:00
|
|
|
"Town RGB House Upstairs Left",
|
|
|
|
"Town RGB House Upstairs Right",
|
2022-10-09 04:13:52 +02:00
|
|
|
"Treehouse Green Bridge 7",
|
|
|
|
"Treehouse Green Bridge Discard",
|
|
|
|
"Shipwreck Discard",
|
|
|
|
"Desert Vault Box",
|
|
|
|
"Mountainside Vault Box",
|
|
|
|
"Mountainside Discard",
|
2023-02-01 21:18:07 +01:00
|
|
|
"Tunnels Theater Flowers EP",
|
|
|
|
"Boat Shipwreck Green EP",
|
2023-07-09 14:21:05 +02:00
|
|
|
"Quarry Stoneworks Control Room Left",
|
2023-12-10 20:36:55 +01:00
|
|
|
]
|
2024-04-12 00:27:42 +02:00
|
|
|
|
2024-01-16 13:14:38 +01:00
|
|
|
# Add Obelisk Sides that contain EPs that are meant to be hinted, if they are necessary to complete the Obelisk Side
|
2024-02-28 04:44:22 +01:00
|
|
|
if "0x33A20" not in world.player_logic.COMPLETELY_DISABLED_ENTITIES:
|
2024-01-16 13:14:38 +01:00
|
|
|
priority.append("Town Obelisk Side 6") # Theater Flowers EP
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
if "0x28B29" not in world.player_logic.COMPLETELY_DISABLED_ENTITIES:
|
2024-01-16 13:14:38 +01:00
|
|
|
priority.append("Treehouse Obelisk Side 4") # Shipwreck Green EP
|
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
if "0x33600" not in world.player_logic.COMPLETELY_DISABLED_ENTITIES:
|
|
|
|
priority.append("Town Obelisk Side 2") # Tutorial Patio Flowers EP.
|
|
|
|
|
2024-01-16 13:14:38 +01:00
|
|
|
return priority
|
2022-10-09 04:13:52 +02:00
|
|
|
|
|
|
|
|
2024-08-20 01:34:40 +02:00
|
|
|
def try_getting_location_group_for_location(world: "WitnessWorld", hint_loc: Location) -> Tuple[str, str]:
|
|
|
|
allow_regions = world.options.vague_hints == "experimental"
|
|
|
|
|
|
|
|
possible_location_groups = {
|
|
|
|
group_name: group_locations
|
|
|
|
for group_name, group_locations in world.multiworld.worlds[hint_loc.player].location_name_groups.items()
|
|
|
|
if hint_loc.name in group_locations
|
|
|
|
}
|
|
|
|
|
|
|
|
locations_in_that_world = {
|
|
|
|
location.name for location in world.multiworld.get_locations(hint_loc.player) if not location.is_event
|
|
|
|
}
|
|
|
|
|
|
|
|
valid_location_groups: Dict[str, int] = {}
|
|
|
|
|
|
|
|
# Find valid location groups.
|
|
|
|
for group, locations in possible_location_groups.items():
|
|
|
|
if group == "Everywhere":
|
|
|
|
continue
|
|
|
|
present_locations = sum(location in locations_in_that_world for location in locations)
|
|
|
|
valid_location_groups[group] = present_locations
|
|
|
|
|
|
|
|
# If there are valid location groups, use a random one.
|
|
|
|
if valid_location_groups:
|
|
|
|
# If there are location groups with more than 1 location, remove any that only have 1.
|
|
|
|
if any(num_locs > 1 for num_locs in valid_location_groups.values()):
|
|
|
|
valid_location_groups = {name: num_locs for name, num_locs in valid_location_groups.items() if num_locs > 1}
|
|
|
|
|
|
|
|
location_groups_with_weights = {
|
|
|
|
# Listen. Just don't worry about it. :)))
|
|
|
|
location_group: (x ** 0.6) * math.e ** (- (x / 7) ** 0.6) if x > 6 else x / 6
|
|
|
|
for location_group, x in valid_location_groups.items()
|
|
|
|
}
|
|
|
|
|
|
|
|
location_groups = list(location_groups_with_weights.keys())
|
|
|
|
weights = list(location_groups_with_weights.values())
|
|
|
|
|
|
|
|
return world.random.choices(location_groups, weights, k=1)[0], "Group"
|
|
|
|
|
|
|
|
if allow_regions:
|
|
|
|
return cast(Region, hint_loc.parent_region).name, "Region"
|
|
|
|
|
|
|
|
return "Everywhere", "Everywhere"
|
|
|
|
|
|
|
|
|
2024-04-12 00:27:42 +02:00
|
|
|
def word_direct_hint(world: "WitnessWorld", hint: WitnessLocationHint) -> WitnessWordedHint:
|
2024-02-28 04:44:22 +01:00
|
|
|
location_name = hint.location.name
|
|
|
|
if hint.location.player != world.player:
|
2024-08-24 02:08:46 +02:00
|
|
|
location_name += " (" + world.player_name + ")"
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
item = hint.location.item
|
2024-07-02 23:59:26 +02:00
|
|
|
|
|
|
|
item_name = "Nothing"
|
|
|
|
if item is not None:
|
|
|
|
item_name = item.name
|
|
|
|
|
|
|
|
if item.player != world.player:
|
2024-08-24 02:08:46 +02:00
|
|
|
item_name += " (" + world.player_name + ")"
|
2024-02-28 04:44:22 +01:00
|
|
|
|
2024-08-20 01:34:40 +02:00
|
|
|
hint_text = ""
|
|
|
|
area: Optional[str] = None
|
|
|
|
|
|
|
|
if world.options.vague_hints:
|
|
|
|
chosen_group, group_type = try_getting_location_group_for_location(world, hint.location)
|
2024-02-28 04:44:22 +01:00
|
|
|
|
2024-08-20 01:34:40 +02:00
|
|
|
if hint.location.player == world.player:
|
|
|
|
area = chosen_group
|
|
|
|
|
|
|
|
# local locations should only ever return a location group, as Witness defines groups for every location.
|
|
|
|
hint_text = f"{item_name} can be found in the {area} area."
|
|
|
|
else:
|
|
|
|
player_name = world.multiworld.get_player_name(hint.location.player)
|
|
|
|
|
|
|
|
if group_type == "Everywhere":
|
|
|
|
location_name = f"a location in {player_name}'s world"
|
|
|
|
elif group_type == "Group":
|
|
|
|
location_name = f"a \"{chosen_group}\" location in {player_name}'s world"
|
|
|
|
elif group_type == "Region":
|
|
|
|
if chosen_group == "Menu":
|
|
|
|
location_name = f"a location near the start of {player_name}'s game (\"Menu\" region)"
|
|
|
|
else:
|
|
|
|
location_name = f"a location in {player_name}'s \"{chosen_group}\" region"
|
|
|
|
|
|
|
|
if hint_text == "":
|
|
|
|
if hint.hint_came_from_location:
|
|
|
|
hint_text = f"{location_name} contains {item_name}."
|
|
|
|
else:
|
|
|
|
hint_text = f"{item_name} can be found at {location_name}."
|
|
|
|
|
|
|
|
return WitnessWordedHint(hint_text, hint.location, area=area, vague_location_hint=bool(world.options.vague_hints))
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
|
2024-07-02 23:59:26 +02:00
|
|
|
def hint_from_item(world: "WitnessWorld", item_name: str,
|
|
|
|
own_itempool: List["WitnessItem"]) -> Optional[WitnessLocationHint]:
|
|
|
|
def get_real_location(multiworld: MultiWorld, location: Location) -> Location:
|
2024-04-18 18:57:22 +02:00
|
|
|
"""If this location is from an item_link pseudo-world, get the location that the item_link item is on.
|
|
|
|
Return the original location otherwise / as a fallback."""
|
|
|
|
if location.player not in world.multiworld.groups:
|
|
|
|
return location
|
|
|
|
|
|
|
|
try:
|
2024-07-02 23:59:26 +02:00
|
|
|
if not location.item:
|
|
|
|
return location
|
2024-04-18 18:57:22 +02:00
|
|
|
return multiworld.find_item(location.item.name, location.player)
|
|
|
|
except StopIteration:
|
|
|
|
return location
|
|
|
|
|
|
|
|
locations = [
|
|
|
|
get_real_location(world.multiworld, item.location)
|
|
|
|
for item in own_itempool if item.name == item_name and item.location
|
|
|
|
]
|
2023-11-24 06:27:03 +01:00
|
|
|
|
|
|
|
if not locations:
|
|
|
|
return None
|
|
|
|
|
|
|
|
location_obj = world.random.choice(locations)
|
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
return WitnessLocationHint(location_obj, False)
|
2022-10-09 04:13:52 +02:00
|
|
|
|
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
def hint_from_location(world: "WitnessWorld", location: str) -> Optional[WitnessLocationHint]:
|
2024-07-02 23:59:26 +02:00
|
|
|
return WitnessLocationHint(world.get_location(location), True)
|
2022-10-09 04:13:52 +02:00
|
|
|
|
|
|
|
|
2024-08-20 01:34:40 +02:00
|
|
|
def get_item_and_location_names_in_random_order(world: "WitnessWorld",
|
|
|
|
own_itempool: List["WitnessItem"]) -> Tuple[List[str], List[str]]:
|
|
|
|
prog_item_names_in_this_world = [
|
2024-02-28 04:44:22 +01:00
|
|
|
item.name for item in own_itempool
|
|
|
|
if item.advancement and item.code and item.location
|
2024-08-20 01:34:40 +02:00
|
|
|
]
|
|
|
|
world.random.shuffle(prog_item_names_in_this_world)
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-08-20 01:34:40 +02:00
|
|
|
locations_in_this_world = [
|
|
|
|
location for location in world.multiworld.get_locations(world.player)
|
|
|
|
if location.item and not location.is_event and location.progress_type != LocationProgressType.EXCLUDED
|
|
|
|
]
|
2024-02-28 04:44:22 +01:00
|
|
|
world.random.shuffle(locations_in_this_world)
|
|
|
|
|
2024-08-20 01:34:40 +02:00
|
|
|
if world.options.vague_hints:
|
|
|
|
locations_in_this_world.sort(key=lambda location: cast(Item, location.item).advancement)
|
|
|
|
|
|
|
|
location_names_in_this_world = [location.name for location in locations_in_this_world]
|
|
|
|
|
|
|
|
return prog_item_names_in_this_world, location_names_in_this_world
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
|
2024-07-02 23:59:26 +02:00
|
|
|
def make_always_and_priority_hints(world: "WitnessWorld", own_itempool: List["WitnessItem"],
|
2024-02-28 04:44:22 +01:00
|
|
|
already_hinted_locations: Set[Location]
|
|
|
|
) -> Tuple[List[WitnessLocationHint], List[WitnessLocationHint]]:
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-08-20 01:34:40 +02:00
|
|
|
prog_items_in_this_world, loc_in_this_world = get_item_and_location_names_in_random_order(world, own_itempool)
|
|
|
|
|
2022-10-09 04:13:52 +02:00
|
|
|
always_items = [
|
2023-11-24 06:27:03 +01:00
|
|
|
item for item in get_always_hint_items(world)
|
2022-10-09 04:13:52 +02:00
|
|
|
if item in prog_items_in_this_world
|
|
|
|
]
|
|
|
|
priority_items = [
|
2023-11-24 06:27:03 +01:00
|
|
|
item for item in get_priority_hint_items(world)
|
2022-10-09 04:13:52 +02:00
|
|
|
if item in prog_items_in_this_world
|
|
|
|
]
|
|
|
|
|
2024-08-20 01:34:40 +02:00
|
|
|
if world.options.vague_hints:
|
|
|
|
always_locations, priority_locations = [], []
|
|
|
|
else:
|
|
|
|
always_locations = [
|
|
|
|
location for location in get_always_hint_locations(world)
|
|
|
|
if location in loc_in_this_world
|
|
|
|
]
|
|
|
|
priority_locations = [
|
|
|
|
location for location in get_priority_hint_locations(world)
|
|
|
|
if location in loc_in_this_world
|
|
|
|
]
|
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
# Get always and priority location/item hints
|
|
|
|
always_location_hints = {hint_from_location(world, location) for location in always_locations}
|
|
|
|
always_item_hints = {hint_from_item(world, item, own_itempool) for item in always_items}
|
|
|
|
priority_location_hints = {hint_from_location(world, location) for location in priority_locations}
|
|
|
|
priority_item_hints = {hint_from_item(world, item, own_itempool) for item in priority_items}
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
# Combine the sets. This will get rid of duplicates
|
|
|
|
always_hints_set = always_item_hints | always_location_hints
|
|
|
|
priority_hints_set = priority_item_hints | priority_location_hints
|
2023-02-01 21:18:07 +01:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
# Make sure priority hints doesn't contain any hints that are already always hints.
|
|
|
|
priority_hints_set -= always_hints_set
|
2023-02-01 21:18:07 +01:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
always_generator = [hint for hint in always_hints_set if hint and hint.location not in already_hinted_locations]
|
|
|
|
priority_generator = [hint for hint in priority_hints_set if hint and hint.location not in already_hinted_locations]
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
# Convert both hint types to list and then shuffle. Also, get rid of None and Tutorial Gate Open.
|
|
|
|
always_hints = sorted(always_generator, key=lambda h: h.location)
|
|
|
|
priority_hints = sorted(priority_generator, key=lambda h: h.location)
|
|
|
|
world.random.shuffle(always_hints)
|
|
|
|
world.random.shuffle(priority_hints)
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
return always_hints, priority_hints
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2023-02-01 21:18:07 +01:00
|
|
|
|
2024-07-02 23:59:26 +02:00
|
|
|
def make_extra_location_hints(world: "WitnessWorld", hint_amount: int, own_itempool: List["WitnessItem"],
|
2024-02-28 04:44:22 +01:00
|
|
|
already_hinted_locations: Set[Location], hints_to_use_first: List[WitnessLocationHint],
|
|
|
|
unhinted_locations_for_hinted_areas: Dict[str, Set[Location]]) -> List[WitnessWordedHint]:
|
2024-08-20 01:34:40 +02:00
|
|
|
prog_items_in_this_world, locations_in_this_world = get_item_and_location_names_in_random_order(world, own_itempool)
|
2023-02-01 21:18:07 +01:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
next_random_hint_is_location = world.random.randrange(0, 2)
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-07-02 23:59:26 +02:00
|
|
|
hints: List[WitnessWordedHint] = []
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
# This is a way to reverse a Dict[a,List[b]] to a Dict[b,a]
|
2024-04-12 00:27:42 +02:00
|
|
|
area_reverse_lookup = {
|
|
|
|
unhinted_location: hinted_area
|
|
|
|
for hinted_area, unhinted_locations in unhinted_locations_for_hinted_areas.items()
|
|
|
|
for unhinted_location in unhinted_locations
|
|
|
|
}
|
2023-11-24 06:27:03 +01:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
while len(hints) < hint_amount:
|
|
|
|
if not prog_items_in_this_world and not locations_in_this_world and not hints_to_use_first:
|
2024-08-24 02:08:46 +02:00
|
|
|
logging.warning(f"Ran out of items/locations to hint for player {world.player_name}.")
|
2024-02-28 04:44:22 +01:00
|
|
|
break
|
|
|
|
|
2024-07-02 23:59:26 +02:00
|
|
|
location_hint: Optional[WitnessLocationHint]
|
2024-02-28 04:44:22 +01:00
|
|
|
if hints_to_use_first:
|
|
|
|
location_hint = hints_to_use_first.pop()
|
|
|
|
elif next_random_hint_is_location and locations_in_this_world:
|
|
|
|
location_hint = hint_from_location(world, locations_in_this_world.pop())
|
|
|
|
elif not next_random_hint_is_location and prog_items_in_this_world:
|
|
|
|
location_hint = hint_from_item(world, prog_items_in_this_world.pop(), own_itempool)
|
|
|
|
# The list that the hint was supposed to be taken from was empty.
|
|
|
|
# Try the other list, which has to still have something, as otherwise, all lists would be empty,
|
|
|
|
# which would have triggered the guard condition above.
|
|
|
|
else:
|
|
|
|
next_random_hint_is_location = not next_random_hint_is_location
|
2023-11-24 06:27:03 +01:00
|
|
|
continue
|
|
|
|
|
2024-07-02 23:59:26 +02:00
|
|
|
if location_hint is None or location_hint.location in already_hinted_locations:
|
2024-02-28 04:44:22 +01:00
|
|
|
continue
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
# Don't hint locations in areas that are almost fully hinted out already
|
|
|
|
if location_hint.location in area_reverse_lookup:
|
|
|
|
area = area_reverse_lookup[location_hint.location]
|
|
|
|
if len(unhinted_locations_for_hinted_areas[area]) == 1:
|
|
|
|
continue
|
|
|
|
del area_reverse_lookup[location_hint.location]
|
|
|
|
unhinted_locations_for_hinted_areas[area] -= {location_hint.location}
|
2023-11-24 06:27:03 +01:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
hints.append(word_direct_hint(world, location_hint))
|
|
|
|
already_hinted_locations.add(location_hint.location)
|
2023-02-01 21:18:07 +01:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
next_random_hint_is_location = not next_random_hint_is_location
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
return hints
|
2022-10-09 04:13:52 +02:00
|
|
|
|
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
def choose_areas(world: "WitnessWorld", amount: int, locations_per_area: Dict[str, List[Location]],
|
|
|
|
already_hinted_locations: Set[Location]) -> Tuple[List[str], Dict[str, Set[Location]]]:
|
|
|
|
"""
|
|
|
|
Choose areas to hint.
|
|
|
|
This takes into account that some areas may already have had items hinted in them through location hints.
|
|
|
|
When this happens, they are made less likely to receive an area hint.
|
|
|
|
"""
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-07-02 23:59:26 +02:00
|
|
|
unhinted_locations_per_area = {}
|
|
|
|
unhinted_location_percentage_per_area = {}
|
2023-11-24 06:27:03 +01:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
for area_name, locations in locations_per_area.items():
|
|
|
|
not_yet_hinted_locations = sum(location not in already_hinted_locations for location in locations)
|
|
|
|
unhinted_locations_per_area[area_name] = {loc for loc in locations if loc not in already_hinted_locations}
|
|
|
|
unhinted_location_percentage_per_area[area_name] = not_yet_hinted_locations / len(locations)
|
2023-06-21 00:45:26 +02:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
items_per_area = {area_name: [location.item for location in locations]
|
|
|
|
for area_name, locations in locations_per_area.items()}
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
areas = sorted(area for area in items_per_area if unhinted_location_percentage_per_area[area])
|
|
|
|
weights = [unhinted_location_percentage_per_area[area] for area in areas]
|
2023-11-24 06:27:03 +01:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
amount = min(amount, len(weights))
|
|
|
|
|
|
|
|
hinted_areas = weighted_sample(world.random, areas, weights, amount)
|
|
|
|
|
|
|
|
return hinted_areas, unhinted_locations_per_area
|
|
|
|
|
|
|
|
|
|
|
|
def get_hintable_areas(world: "WitnessWorld") -> Tuple[Dict[str, List[Location]], Dict[str, List[Item]]]:
|
2024-04-12 00:27:42 +02:00
|
|
|
potential_areas = list(static_witness_logic.ALL_AREAS_BY_NAME.keys())
|
2024-02-28 04:44:22 +01:00
|
|
|
|
2024-07-02 23:59:26 +02:00
|
|
|
locations_per_area = {}
|
|
|
|
items_per_area = {}
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
for area in potential_areas:
|
|
|
|
regions = [
|
2024-06-01 23:11:28 +02:00
|
|
|
world.get_region(region)
|
2024-04-12 00:27:42 +02:00
|
|
|
for region in static_witness_logic.ALL_AREAS_BY_NAME[area]["regions"]
|
2024-06-01 23:11:28 +02:00
|
|
|
if region in world.player_regions.created_region_names
|
2024-02-28 04:44:22 +01:00
|
|
|
]
|
2024-08-20 01:34:40 +02:00
|
|
|
locations = [location for region in regions for location in region.get_locations() if not location.is_event]
|
2023-11-24 06:27:03 +01:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
if locations:
|
|
|
|
locations_per_area[area] = locations
|
|
|
|
items_per_area[area] = [location.item for location in locations]
|
2023-11-24 06:27:03 +01:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
return locations_per_area, items_per_area
|
|
|
|
|
|
|
|
|
2024-08-20 01:16:35 +02:00
|
|
|
def word_area_hint(world: "WitnessWorld", hinted_area: str, area_items: List[Item]) -> Tuple[str, int, Optional[int]]:
|
2024-02-28 04:44:22 +01:00
|
|
|
"""
|
|
|
|
Word the hint for an area using natural sounding language.
|
|
|
|
This takes into account how much progression there is, how much of it is local/non-local, and whether there are
|
|
|
|
any local lasers to be found in this area.
|
|
|
|
"""
|
|
|
|
|
2024-08-20 01:16:35 +02:00
|
|
|
local_progression = sum(item.player == world.player and item.advancement for item in area_items)
|
|
|
|
non_local_progression = sum(item.player != world.player and item.advancement for item in area_items)
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
laser_names = {"Symmetry Laser", "Desert Laser", "Quarry Laser", "Shadows Laser", "Town Laser", "Monastery Laser",
|
|
|
|
"Jungle Laser", "Bunker Laser", "Swamp Laser", "Treehouse Laser", "Keep Laser", }
|
|
|
|
|
|
|
|
local_lasers = sum(
|
|
|
|
item.player == world.player and item.name in laser_names
|
2024-08-20 01:16:35 +02:00
|
|
|
for item in area_items
|
2024-02-28 04:44:22 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
total_progression = non_local_progression + local_progression
|
|
|
|
|
|
|
|
player_count = world.multiworld.players
|
|
|
|
|
|
|
|
area_progression_word = "Both" if total_progression == 2 else "All"
|
|
|
|
|
2024-08-20 01:16:35 +02:00
|
|
|
hint_string = f"In the {hinted_area} area, you will find "
|
|
|
|
|
|
|
|
hunt_panels = None
|
|
|
|
if world.options.victory_condition == "panel_hunt":
|
|
|
|
hunt_panels = sum(
|
|
|
|
static_witness_logic.ENTITIES_BY_HEX[hunt_entity]["area"]["name"] == hinted_area
|
|
|
|
for hunt_entity in world.player_logic.HUNT_ENTITIES
|
|
|
|
)
|
|
|
|
|
|
|
|
if not hunt_panels:
|
|
|
|
hint_string += "no Hunt Panels and "
|
|
|
|
|
|
|
|
elif hunt_panels == 1:
|
|
|
|
hint_string += "1 Hunt Panel and "
|
|
|
|
|
|
|
|
else:
|
|
|
|
hint_string += f"{hunt_panels} Hunt Panels and "
|
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
if not total_progression:
|
2024-08-20 01:16:35 +02:00
|
|
|
hint_string += "no progression items."
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
elif total_progression == 1:
|
2024-08-20 01:16:35 +02:00
|
|
|
hint_string += "1 progression item."
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
if player_count > 1:
|
|
|
|
if local_lasers:
|
|
|
|
hint_string += "\nThis item is a laser for this world."
|
|
|
|
elif non_local_progression:
|
|
|
|
other_player_str = "the other player" if player_count == 2 else "another player"
|
|
|
|
hint_string += f"\nThis item is for {other_player_str}."
|
|
|
|
else:
|
|
|
|
hint_string += "\nThis item is for this world."
|
2022-10-09 04:13:52 +02:00
|
|
|
else:
|
2024-02-28 04:44:22 +01:00
|
|
|
if local_lasers:
|
|
|
|
hint_string += "\nThis item is a laser."
|
|
|
|
|
|
|
|
else:
|
2024-08-20 01:16:35 +02:00
|
|
|
hint_string += f"{total_progression} progression items."
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
if local_lasers == total_progression:
|
|
|
|
sentence_end = (" for this world." if player_count > 1 else ".")
|
2024-04-12 00:27:42 +02:00
|
|
|
hint_string += "\nAll of them are lasers" + sentence_end
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
elif player_count > 1:
|
|
|
|
if local_progression and non_local_progression:
|
|
|
|
if non_local_progression == 1:
|
|
|
|
other_player_str = "the other player" if player_count == 2 else "another player"
|
|
|
|
hint_string += f"\nOne of them is for {other_player_str}."
|
|
|
|
else:
|
|
|
|
other_player_str = "the other player" if player_count == 2 else "other players"
|
|
|
|
hint_string += f"\n{non_local_progression} of them are for {other_player_str}."
|
|
|
|
elif non_local_progression:
|
|
|
|
other_players_str = "the other player" if player_count == 2 else "other players"
|
|
|
|
hint_string += f"\n{area_progression_word} of them are for {other_players_str}."
|
|
|
|
elif local_progression:
|
|
|
|
hint_string += f"\n{area_progression_word} of them are for this world."
|
|
|
|
|
|
|
|
if local_lasers == 1:
|
|
|
|
if not non_local_progression:
|
|
|
|
hint_string += "\nAlso, one of them is a laser."
|
|
|
|
else:
|
|
|
|
hint_string += "\nAlso, one of them is a laser for this world."
|
|
|
|
elif local_lasers:
|
|
|
|
if not non_local_progression:
|
|
|
|
hint_string += f"\nAlso, {local_lasers} of them are lasers."
|
|
|
|
else:
|
|
|
|
hint_string += f"\nAlso, {local_lasers} of them are lasers for this world."
|
2023-11-24 06:27:03 +01:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
else:
|
|
|
|
if local_lasers == 1:
|
|
|
|
hint_string += "\nOne of them is a laser."
|
|
|
|
elif local_lasers:
|
|
|
|
hint_string += f"\n{local_lasers} of them are lasers."
|
2023-11-24 06:27:03 +01:00
|
|
|
|
2024-08-20 01:16:35 +02:00
|
|
|
return hint_string, total_progression, hunt_panels
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2023-11-24 06:27:03 +01:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
def make_area_hints(world: "WitnessWorld", amount: int, already_hinted_locations: Set[Location]
|
|
|
|
) -> Tuple[List[WitnessWordedHint], Dict[str, Set[Location]]]:
|
|
|
|
locs_per_area, items_per_area = get_hintable_areas(world)
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
hinted_areas, unhinted_locations_per_area = choose_areas(world, amount, locs_per_area, already_hinted_locations)
|
|
|
|
|
|
|
|
hints = []
|
|
|
|
|
|
|
|
for hinted_area in hinted_areas:
|
2024-08-20 01:16:35 +02:00
|
|
|
hint_string, prog_amount, hunt_panels = word_area_hint(world, hinted_area, items_per_area[hinted_area])
|
2024-02-28 04:44:22 +01:00
|
|
|
|
2024-08-20 01:16:35 +02:00
|
|
|
hints.append(WitnessWordedHint(hint_string, None, f"hinted_area:{hinted_area}", prog_amount, hunt_panels))
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
if len(hinted_areas) < amount:
|
2024-08-24 02:08:46 +02:00
|
|
|
logging.warning(f"Was not able to make {amount} area hints for player {world.player_name}. "
|
2024-02-28 04:44:22 +01:00
|
|
|
f"Made {len(hinted_areas)} instead, and filled the rest with random location hints.")
|
|
|
|
|
|
|
|
return hints, unhinted_locations_per_area
|
|
|
|
|
|
|
|
|
2024-03-05 22:54:02 +01:00
|
|
|
def create_all_hints(world: "WitnessWorld", hint_amount: int, area_hints: int,
|
|
|
|
already_hinted_locations: Set[Location]) -> List[WitnessWordedHint]:
|
2024-02-28 04:44:22 +01:00
|
|
|
generated_hints: List[WitnessWordedHint] = []
|
|
|
|
|
|
|
|
state = CollectionState(world.multiworld)
|
|
|
|
|
|
|
|
# Keep track of already hinted locations. Consider early Tutorial as "already hinted"
|
|
|
|
|
2024-03-05 22:54:02 +01:00
|
|
|
already_hinted_locations |= {
|
2024-02-28 04:44:22 +01:00
|
|
|
loc for loc in world.multiworld.get_reachable_locations(state, world.player)
|
2024-04-12 00:27:42 +02:00
|
|
|
if loc.address and static_witness_logic.ENTITIES_BY_NAME[loc.name]["area"]["name"] == "Tutorial (Inside)"
|
2024-02-28 04:44:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
intended_location_hints = hint_amount - area_hints
|
|
|
|
|
|
|
|
# First, make always and priority hints.
|
|
|
|
|
|
|
|
always_hints, priority_hints = make_always_and_priority_hints(
|
|
|
|
world, world.own_itempool, already_hinted_locations
|
|
|
|
)
|
|
|
|
|
|
|
|
generated_always_hints = len(always_hints)
|
|
|
|
possible_priority_hints = len(priority_hints)
|
|
|
|
|
|
|
|
# Make as many always hints as possible
|
|
|
|
always_hints_to_use = min(intended_location_hints, generated_always_hints)
|
|
|
|
|
|
|
|
# Make up to half of the rest of the location hints priority hints, using up to half of the possibly priority hints
|
|
|
|
remaining_location_hints = intended_location_hints - always_hints_to_use
|
|
|
|
priority_hints_to_use = int(max(0.0, min(possible_priority_hints / 2, remaining_location_hints / 2)))
|
|
|
|
|
|
|
|
for _ in range(always_hints_to_use):
|
|
|
|
location_hint = always_hints.pop()
|
|
|
|
generated_hints.append(word_direct_hint(world, location_hint))
|
|
|
|
already_hinted_locations.add(location_hint.location)
|
|
|
|
|
|
|
|
for _ in range(priority_hints_to_use):
|
|
|
|
location_hint = priority_hints.pop()
|
|
|
|
generated_hints.append(word_direct_hint(world, location_hint))
|
|
|
|
already_hinted_locations.add(location_hint.location)
|
|
|
|
|
|
|
|
location_hints_created_in_round_1 = len(generated_hints)
|
|
|
|
|
2024-07-02 23:59:26 +02:00
|
|
|
unhinted_locations_per_area: Dict[str, Set[Location]] = {}
|
2024-02-28 04:44:22 +01:00
|
|
|
|
|
|
|
# Then, make area hints.
|
|
|
|
if area_hints:
|
|
|
|
generated_area_hints, unhinted_locations_per_area = make_area_hints(world, area_hints, already_hinted_locations)
|
|
|
|
generated_hints += generated_area_hints
|
|
|
|
|
|
|
|
# If we don't have enough hints yet, recalculate always and priority hints, then fill with random hints
|
|
|
|
if len(generated_hints) < hint_amount:
|
|
|
|
remaining_needed_location_hints = hint_amount - len(generated_hints)
|
|
|
|
|
|
|
|
# Save old values for used always and priority hints for later calculations
|
|
|
|
amt_of_used_always_hints = always_hints_to_use
|
|
|
|
amt_of_used_priority_hints = priority_hints_to_use
|
|
|
|
|
|
|
|
# Recalculate how many always hints and priority hints are supposed to be used
|
|
|
|
intended_location_hints = remaining_needed_location_hints + location_hints_created_in_round_1
|
|
|
|
|
|
|
|
always_hints_to_use = min(intended_location_hints, generated_always_hints)
|
|
|
|
priority_hints_to_use = int(max(0.0, min(possible_priority_hints / 2, remaining_location_hints / 2)))
|
|
|
|
|
|
|
|
# If we now need more always hints and priority hints than we thought previously, make some more.
|
|
|
|
more_always_hints = always_hints_to_use - amt_of_used_always_hints
|
|
|
|
more_priority_hints = priority_hints_to_use - amt_of_used_priority_hints
|
|
|
|
|
|
|
|
extra_always_and_priority_hints: List[WitnessLocationHint] = []
|
|
|
|
|
|
|
|
for _ in range(more_always_hints):
|
|
|
|
extra_always_and_priority_hints.append(always_hints.pop())
|
|
|
|
|
|
|
|
for _ in range(more_priority_hints):
|
|
|
|
extra_always_and_priority_hints.append(priority_hints.pop())
|
|
|
|
|
|
|
|
generated_hints += make_extra_location_hints(
|
|
|
|
world, hint_amount - len(generated_hints), world.own_itempool, already_hinted_locations,
|
|
|
|
extra_always_and_priority_hints, unhinted_locations_per_area
|
|
|
|
)
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
# If we still don't have enough for whatever reason, throw a warning, proceed with the lower amount
|
|
|
|
if len(generated_hints) != hint_amount:
|
2024-08-24 02:08:46 +02:00
|
|
|
logging.warning(f"Couldn't generate {hint_amount} hints for player {world.player_name}. "
|
2024-02-28 04:44:22 +01:00
|
|
|
f"Generated {len(generated_hints)} instead.")
|
2022-10-09 04:13:52 +02:00
|
|
|
|
2024-02-28 04:44:22 +01:00
|
|
|
return generated_hints
|
2024-03-05 22:54:02 +01:00
|
|
|
|
|
|
|
|
2024-08-20 01:16:35 +02:00
|
|
|
def get_compact_hint_args(hint: WitnessWordedHint, local_player_number: int) -> CompactHintArgs:
|
|
|
|
"""
|
|
|
|
Arg reference:
|
|
|
|
|
|
|
|
Area Hint: 1st Arg is the amount of area progression and hunt panels. 2nd Arg is the name of the area.
|
|
|
|
Location Hint: 1st Arg is the location's address, second arg is the player number the location belongs to.
|
|
|
|
Junk Hint: 1st Arg is -1, second arg is this slot's player number.
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Is Area Hint
|
2024-08-20 01:34:40 +02:00
|
|
|
if hint.area_amount is not None:
|
2024-08-20 01:16:35 +02:00
|
|
|
area_amount = hint.area_amount
|
|
|
|
hunt_panels = hint.area_hunt_panels
|
|
|
|
|
|
|
|
area_and_hunt_panels = area_amount
|
|
|
|
# Encode amounts together
|
|
|
|
if hunt_panels:
|
|
|
|
area_and_hunt_panels += 0x100 * hunt_panels
|
|
|
|
|
|
|
|
return hint.area, area_and_hunt_panels
|
|
|
|
|
2024-03-05 22:54:02 +01:00
|
|
|
location = hint.location
|
|
|
|
|
2024-08-20 01:16:35 +02:00
|
|
|
# Is location hint
|
2024-07-02 23:59:26 +02:00
|
|
|
if location and location.address is not None:
|
2024-08-20 01:34:40 +02:00
|
|
|
if hint.vague_location_hint and location.player == local_player_number:
|
|
|
|
assert hint.area is not None # A local vague location hint should have an area argument
|
|
|
|
return location.address, "containing_area:" + hint.area
|
|
|
|
else:
|
|
|
|
return location.address, location.player # Scouting does not matter for other players (currently)
|
2024-08-20 01:16:35 +02:00
|
|
|
|
|
|
|
# Is junk / undefined hint
|
|
|
|
return -1, local_player_number
|
|
|
|
|
2024-03-05 22:54:02 +01:00
|
|
|
|
2024-08-20 01:16:35 +02:00
|
|
|
def make_compact_hint_data(hint: WitnessWordedHint, local_player_number: int) -> CompactHintData:
|
|
|
|
compact_arg_1, compact_arg_2 = get_compact_hint_args(hint, local_player_number)
|
|
|
|
return hint.wording, compact_arg_1, compact_arg_2
|
2024-03-05 22:54:02 +01:00
|
|
|
|
|
|
|
|
|
|
|
def make_laser_hints(world: "WitnessWorld", laser_names: List[str]) -> Dict[str, WitnessWordedHint]:
|
2024-07-02 23:59:26 +02:00
|
|
|
laser_hints_by_name = {}
|
2024-03-05 22:54:02 +01:00
|
|
|
|
|
|
|
for item_name in laser_names:
|
|
|
|
location_hint = hint_from_item(world, item_name, world.own_itempool)
|
|
|
|
if not location_hint:
|
|
|
|
continue
|
|
|
|
|
|
|
|
laser_hints_by_name[item_name] = word_direct_hint(world, location_hint)
|
|
|
|
|
|
|
|
return laser_hints_by_name
|