mirror of
				https://github.com/MarioSpore/Grinch-AP.git
				synced 2025-10-21 20:21:32 -06:00 
			
		
		
		
	
		
			
	
	
		
			282 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
		
		
			
		
	
	
			282 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
|   | from BaseClasses import MultiWorld | ||
|  | from .Options import is_option_enabled, get_option_value | ||
|  | 
 | ||
|  | joke_hints = [ | ||
|  |     ("Quaternions", "break", "my brain"), | ||
|  |     ("Eclipse", "has nothing", "but you should do it anyway"), | ||
|  |     ("", "Beep", ""), | ||
|  |     ("Putting in custom subtitles", "shouldn't have been", "as hard as it was..."), | ||
|  |     ("BK mode", "is right", "around the corner"), | ||
|  |     ("", "You can do it!", ""), | ||
|  |     ("", "I believe in you!", ""), | ||
|  |     ("The person playing", "is", "cute <3"), | ||
|  |     ("dash dot, dash dash dash", "dash, dot dot dot dot, dot dot", "dash dot, dash dash dot"), | ||
|  |     ("When you think about it,", "there are actually a lot of", "bubbles in a stream"), | ||
|  |     ("Never gonna give you up", "Never gonna let you down", "Never gonna run around and desert you"), | ||
|  |     ("Thanks to", "the Archipelago developers", "for making this possible."), | ||
|  |     ("Have you tried ChecksFinder?", "If you like puzzles,", "you might enjoy it!"), | ||
|  |     ("Have you tried Dark Souls III?", "A tough game like this", "feels better when friends are helping you!"), | ||
|  |     ("Have you tried Donkey Kong Country 3?", "A legendary game", "from a golden age of platformers!"), | ||
|  |     ("Have you tried Factorio?", "Alone in an unknown world.", "Sound familiar?"), | ||
|  |     ("Have you tried Final Fantasy?", "Experience a classic game", "improved to fit modern standards!"), | ||
|  |     ("Have you tried Hollow Knight?", "Another independent hit", "revolutionising a genre!"), | ||
|  |     ("Have you tried A Link to the Past?", "The Archipelago game", "that started it all!"), | ||
|  |     ("Have you tried Meritous?", "You should know that obscure games", "are often groundbreaking!"), | ||
|  |     ("Have you tried Ocarine of Time?", "One of the biggest randomizers,", "big inspiration for this one's features!"), | ||
|  |     ("Have you tried Raft?", "Haven't you always wanted to explore", "the ocean surrounding this island?"), | ||
|  |     ("Have you tried Risk of Rain 2?", "I haven't either.", "But I hear it's incredible!"), | ||
|  |     ("Have you tried Rogue Legacy?", "After solving so many puzzles", "it's the perfect way to rest your brain."), | ||
|  |     ("Have you tried Secret of Evermore?", "I haven't either", "But I hear it's great!"), | ||
|  |     ("Have you tried Slay the Spire?", "Experience the thrill of combat", "without needing fast fingers!"), | ||
|  |     ("Have you tried SMZ3?", "Why play one incredible game", "when you can play 2 at once?"), | ||
|  |     ("Have you tried Starcraft 2?", "Use strategy and management", "to crush your enemies!"), | ||
|  |     ("Have you tried Super Mario 64?", "3-dimensional games like this", "owe everything to that game."), | ||
|  |     ("Have you tried Super Metroid?", "A classic game", "that started a whole genre."), | ||
|  |     ("Have you tried Timespinner?", "Everyone who plays it", "ends up loving it!"), | ||
|  |     ("Have you tried VVVVVV?", "Experience the essence of gaming", "distilled into its purest form!"), | ||
|  |     ("Have you tried The Witness?", "Oh. I guess you already have.", " Thanks for playing!"), | ||
|  |     ("One day I was fascinated", "by the subject of", "generation of waves by wind"), | ||
|  |     ("I don't like sandwiches", "Why would you think I like sandwiches?", "Have you ever seen me with a sandwich?"), | ||
|  |     ("Where are you right now?", "I'm at soup!", "What do you mean you're at soup?"), | ||
|  |     ("Remember to ask", "in the Archipelago Discord", "what the Functioning Brain does."), | ||
|  |     ("", "Don't use your puzzle skips", "you might need them later"), | ||
|  |     ("", "For an extra challenge", "Try playing blindfolded"), | ||
|  |     ("Go to the top of the mountain", "and see if you can see", "your house"), | ||
|  |     ("Yellow = Red + Green", "Cyan = Green + Blue", "Magenta = Red + Blue"), | ||
|  |     ("", "Maybe that panel really is unsolvable", ""), | ||
|  |     ("", "Did you make sure it was plugged in?", ""), | ||
|  |     ("", "Do not look into laser with remaining eye", ""), | ||
|  |     ("", "Try pressing Space to jump", ""), | ||
|  |     ("The Witness is a Doom clone.", "Just replace the demons", "with puzzles"), | ||
|  |     ("", "Test Hint please ignore", ""), | ||
|  |     ("Shapers can never be placed", "outside the panel boundaries", "even if subtracted."), | ||
|  |     ("", "The Keep laser panels use", "the same trick on both sides!"), | ||
|  |     ("Can't get past a door? Try going around.", "Can't go around? Try building a", "nether portal."), | ||
|  |     ("", "We've been trying to reach you", "about your car's extended warranty"), | ||
|  |     ("I hate this game. I hate this game.", "I hate this game.", "-chess player Bobby Fischer"), | ||
|  |     ("Dear Mario,", "Please come to the castle.", "I've baked a cake for you!"), | ||
|  |     ("Have you tried waking up?", "", "Yeah, me neither."), | ||
|  |     ("Why do they call it The Witness,", "when wit game the player view", "play of with the game."), | ||
|  |     ("", "THE WIND FISH IN NAME ONLY", "FOR IT IS NEITHER"), | ||
|  |     ("Like this game? Try The Wit.nes,", "Understand, INSIGHT, Taiji", "What the Witness?, and Tametsi."), | ||
|  |     ("", "In a race", "It's survival of the Witnesst"), | ||
|  |     ("", "This hint has been removed", "We apologize for your inconvenience."), | ||
|  |     ("", "O-----------", ""), | ||
|  |     ("Circle is draw", "Square is separate", "Line is win"), | ||
|  |     ("Circle is draw", "Star is pair", "Line is win"), | ||
|  |     ("Circle is draw", "Circle is copy", "Line is win"), | ||
|  |     ("Circle is draw", "Dot is eat", "Line is win"), | ||
|  |     ("Circle is start", "Walk is draw", "Line is win"), | ||
|  |     ("Circle is start", "Line is win", "Witness is you"), | ||
|  |     ("Can't find any items?", "Consider a relaxing boat trip", "around the island"), | ||
|  |     ("", "Don't forget to like, comment, and subscribe", ""), | ||
|  |     ("Ah crap, gimme a second.", "[papers rustling]", "Sorry, nothing."), | ||
|  |     ("", "Trying to get a hint?", "Too bad."), | ||
|  |     ("", "Here's a hint:", "Get good at the game."), | ||
|  |     ("", "I'm still not entirely sure", "what we're witnessing here."), | ||
|  |     ("Have you found a red page yet?", "No?", "Then have you found a blue page?"), | ||
|  |     ( | ||
|  |         "And here we see the Witness player,", | ||
|  |         "seeking answers where there are none-", | ||
|  |         "Did someone turn on the loudspeaker?" | ||
|  |     ), | ||
|  |     ( | ||
|  |         "Hints suggested by:", | ||
|  |         "IHNN, Beaker, MrPokemon11, Ember, TheM8, NewSoupVi,", | ||
|  |         "KF, Yoshi348, Berserker, BowlinJim, oddGarrett, Pink Switch." | ||
|  |     ), | ||
|  | ] | ||
|  | 
 | ||
|  | 
 | ||
|  | def get_always_hint_items(world: MultiWorld, player: int): | ||
|  |     priority = [ | ||
|  |         "Boat", | ||
|  |         "Mountain Bottom Floor Final Room Entry (Door)", | ||
|  |         "Caves Mountain Shortcut (Door)", | ||
|  |         "Caves Swamp Shortcut (Door)", | ||
|  |         "Caves Exits to Main Island", | ||
|  |     ] | ||
|  | 
 | ||
|  |     difficulty = get_option_value(world, player, "puzzle_randomization") | ||
|  |     discards = is_option_enabled(world, player, "shuffle_discards") | ||
|  | 
 | ||
|  |     if discards: | ||
|  |         if difficulty == 1: | ||
|  |             priority.append("Arrows") | ||
|  |         else: | ||
|  |             priority.append("Triangles") | ||
|  | 
 | ||
|  |     return priority | ||
|  | 
 | ||
|  | 
 | ||
|  | def get_always_hint_locations(world: MultiWorld, player: int): | ||
|  |     return { | ||
|  |         "Swamp Purple Underwater", | ||
|  |         "Shipwreck Vault Box", | ||
|  |         "Challenge Vault Box", | ||
|  |         "Mountain Bottom Floor Discard", | ||
|  |     } | ||
|  | 
 | ||
|  | 
 | ||
|  | def get_priority_hint_items(world: MultiWorld, player: int): | ||
|  |     priority = { | ||
|  |         "Negative Shapers", | ||
|  |         "Sound Dots", | ||
|  |         "Colored Dots", | ||
|  |         "Stars + Same Colored Symbol", | ||
|  |         "Swamp Entry (Panel)", | ||
|  |         "Swamp Laser Shortcut (Door)", | ||
|  |     } | ||
|  | 
 | ||
|  |     if is_option_enabled(world, player, "shuffle_lasers"): | ||
|  |         lasers = { | ||
|  |             "Symmetry Laser", | ||
|  |             "Desert Laser", | ||
|  |             "Town Laser", | ||
|  |             "Keep Laser", | ||
|  |             "Swamp Laser", | ||
|  |             "Treehouse Laser", | ||
|  |             "Monastery Laser", | ||
|  |             "Jungle Laser", | ||
|  |             "Quarry Laser", | ||
|  |             "Bunker Laser", | ||
|  |             "Shadows Laser", | ||
|  |         } | ||
|  | 
 | ||
|  |         if get_option_value(world, player, "doors") >= 2: | ||
|  |             priority.add("Desert Laser") | ||
|  |             lasers.remove("Desert Laser") | ||
|  |             priority.update(world.random.sample(lasers, 2)) | ||
|  | 
 | ||
|  |         else: | ||
|  |             priority.update(world.random.sample(lasers, 3)) | ||
|  | 
 | ||
|  |     return priority | ||
|  | 
 | ||
|  | 
 | ||
|  | def get_priority_hint_locations(world: MultiWorld, player: int): | ||
|  |     return { | ||
|  |         "Town RGB Room Left", | ||
|  |         "Town RGB Room Right", | ||
|  |         "Treehouse Green Bridge 7", | ||
|  |         "Treehouse Green Bridge Discard", | ||
|  |         "Shipwreck Discard", | ||
|  |         "Desert Vault Box", | ||
|  |         "Mountainside Vault Box", | ||
|  |         "Mountainside Discard", | ||
|  |     } | ||
|  | 
 | ||
|  | 
 | ||
|  | def make_hint_from_item(world: MultiWorld, player: int, item: str): | ||
|  |     location_obj = world.find_item(item, player).item.location | ||
|  |     location_name = location_obj.name | ||
|  |     if location_obj.player != player: | ||
|  |         location_name += " (" + world.get_player_name(location_obj.player) + ")" | ||
|  | 
 | ||
|  |     return location_name, item, location_obj.address if(location_obj.player == player) else -1 | ||
|  | 
 | ||
|  | 
 | ||
|  | def make_hint_from_location(world: MultiWorld, player: int, location: str): | ||
|  |     location_obj = world.get_location(location, player) | ||
|  |     item_obj = world.get_location(location, player).item | ||
|  |     item_name = item_obj.name | ||
|  |     if item_obj.player != player: | ||
|  |         item_name += " (" + world.get_player_name(item_obj.player) + ")" | ||
|  | 
 | ||
|  |     return location, item_name, location_obj.address if(location_obj.player == player) else -1 | ||
|  | 
 | ||
|  | 
 | ||
|  | def make_hints(world: MultiWorld, player: int, hint_amount: int): | ||
|  |     hints = list() | ||
|  | 
 | ||
|  |     prog_items_in_this_world = { | ||
|  |         item.name for item in world.get_items() | ||
|  |         if item.player == player and item.code and item.advancement | ||
|  |     } | ||
|  |     loc_in_this_world = { | ||
|  |         location.name for location in world.get_locations() | ||
|  |         if location.player == player and not location.event | ||
|  |     } | ||
|  | 
 | ||
|  |     always_locations = [ | ||
|  |         location for location in get_always_hint_locations(world, player) | ||
|  |         if location in loc_in_this_world | ||
|  |     ] | ||
|  |     always_items = [ | ||
|  |         item for item in get_always_hint_items(world, player) | ||
|  |         if item in prog_items_in_this_world | ||
|  |     ] | ||
|  |     priority_locations = [ | ||
|  |         location for location in get_priority_hint_locations(world, player) | ||
|  |         if location in loc_in_this_world | ||
|  |     ] | ||
|  |     priority_items = [ | ||
|  |         item for item in get_priority_hint_items(world, player) | ||
|  |         if item in prog_items_in_this_world | ||
|  |     ] | ||
|  | 
 | ||
|  |     always_hint_pairs = dict() | ||
|  | 
 | ||
|  |     for item in always_items: | ||
|  |         hint_pair = make_hint_from_item(world, player, item) | ||
|  |         always_hint_pairs[hint_pair[0]] = (hint_pair[1], True, hint_pair[2]) | ||
|  | 
 | ||
|  |     for location in always_locations: | ||
|  |         hint_pair = make_hint_from_location(world, player, location) | ||
|  |         always_hint_pairs[hint_pair[0]] = (hint_pair[1], False, hint_pair[2]) | ||
|  | 
 | ||
|  |     priority_hint_pairs = dict() | ||
|  | 
 | ||
|  |     for item in priority_items: | ||
|  |         hint_pair = make_hint_from_item(world, player, item) | ||
|  |         priority_hint_pairs[hint_pair[0]] = (hint_pair[1], True, hint_pair[2]) | ||
|  | 
 | ||
|  |     for location in priority_locations: | ||
|  |         hint_pair = make_hint_from_location(world, player, location) | ||
|  |         priority_hint_pairs[hint_pair[0]] = (hint_pair[1], False, hint_pair[2]) | ||
|  | 
 | ||
|  |     for loc, item in always_hint_pairs.items(): | ||
|  |         if item[1]: | ||
|  |             hints.append((item[0], "can be found at", loc, item[2])) | ||
|  |         else: | ||
|  |             hints.append((loc, "contains", item[0], item[2])) | ||
|  | 
 | ||
|  |     next_random_hint_is_item = world.random.randint(0, 2) | ||
|  | 
 | ||
|  |     prog_items_in_this_world = sorted(list(prog_items_in_this_world)) | ||
|  |     locations_in_this_world = sorted(list(loc_in_this_world)) | ||
|  | 
 | ||
|  |     world.random.shuffle(prog_items_in_this_world) | ||
|  |     world.random.shuffle(locations_in_this_world) | ||
|  | 
 | ||
|  |     while len(hints) < hint_amount: | ||
|  |         if priority_hint_pairs: | ||
|  |             loc = world.random.choice(list(priority_hint_pairs.keys())) | ||
|  |             item = priority_hint_pairs[loc] | ||
|  |             del priority_hint_pairs[loc] | ||
|  | 
 | ||
|  |             if item[1]: | ||
|  |                 hints.append((item[0], "can be found at", loc, item[2])) | ||
|  |             else: | ||
|  |                 hints.append((loc, "contains", item[0], item[2])) | ||
|  |             continue | ||
|  | 
 | ||
|  |         if next_random_hint_is_item: | ||
|  |             if not prog_items_in_this_world: | ||
|  |                 next_random_hint_is_item = not next_random_hint_is_item | ||
|  |                 continue | ||
|  | 
 | ||
|  |             hint = make_hint_from_item(world, player, prog_items_in_this_world.pop()) | ||
|  |             hints.append((hint[1], "can be found at", hint[0], hint[2])) | ||
|  |         else: | ||
|  |             hint = make_hint_from_location(world, player, locations_in_this_world.pop()) | ||
|  |             hints.append((hint[0], "contains", hint[1], hint[2])) | ||
|  | 
 | ||
|  |         next_random_hint_is_item = not next_random_hint_is_item | ||
|  | 
 | ||
|  |     return hints | ||
|  | 
 | ||
|  | 
 | ||
|  | def generate_joke_hints(world: MultiWorld, amount: int): | ||
|  |     return [(x, y, z, -1) for (x, y, z) in world.random.sample(joke_hints, amount)] |