| 
									
										
										
										
											2024-07-23 03:04:24 -04:00
										 |  |  | from typing import Dict, List, Any, Tuple, TypedDict, ClassVar, Union | 
					
						
							| 
									
										
										
										
											2024-04-13 18:20:52 -04:00
										 |  |  | from logging import warning | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  | from BaseClasses import Region, Location, Item, Tutorial, ItemClassification, MultiWorld | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | from .items import item_name_to_id, item_table, item_name_groups, fool_tiers, filler_items, slot_data_item_names | 
					
						
							|  |  |  | from .locations import location_table, location_name_groups, location_name_to_id, hexagon_locations | 
					
						
							|  |  |  | from .rules import set_location_rules, set_region_rules, randomize_ability_unlocks, gold_hexagon | 
					
						
							|  |  |  | from .er_rules import set_er_location_rules | 
					
						
							|  |  |  | from .regions import tunic_regions | 
					
						
							|  |  |  | from .er_scripts import create_er_regions | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  | from .er_data import portal_mapping | 
					
						
							| 
									
										
										
										
											2024-06-03 04:44:37 -04:00
										 |  |  | from .options import TunicOptions, EntranceRando, tunic_option_groups, tunic_option_presets, TunicPlandoConnections | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | from worlds.AutoWorld import WebWorld, World | 
					
						
							| 
									
										
										
										
											2024-06-01 06:34:41 -05:00
										 |  |  | from Options import PlandoConnection | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | from decimal import Decimal, ROUND_HALF_UP | 
					
						
							| 
									
										
										
										
											2024-07-23 03:04:24 -04:00
										 |  |  | from settings import Group, Bool | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TunicSettings(Group): | 
					
						
							|  |  |  |     class DisableLocalSpoiler(Bool): | 
					
						
							|  |  |  |         """Disallows the TUNIC client from creating a local spoiler log.""" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     disable_local_spoiler: Union[DisableLocalSpoiler, bool] = False | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TunicWeb(WebWorld): | 
					
						
							|  |  |  |     tutorials = [ | 
					
						
							|  |  |  |         Tutorial( | 
					
						
							|  |  |  |             tutorial_name="Multiworld Setup Guide", | 
					
						
							|  |  |  |             description="A guide to setting up the TUNIC Randomizer for Archipelago multiworld games.", | 
					
						
							|  |  |  |             language="English", | 
					
						
							|  |  |  |             file_name="setup_en.md", | 
					
						
							|  |  |  |             link="setup/en", | 
					
						
							|  |  |  |             authors=["SilentDestroyer"] | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |     ] | 
					
						
							|  |  |  |     theme = "grassFlowers" | 
					
						
							| 
									
										
										
										
											2024-01-17 19:56:34 -05:00
										 |  |  |     game = "TUNIC" | 
					
						
							| 
									
										
										
										
											2024-05-21 18:12:52 -04:00
										 |  |  |     option_groups = tunic_option_groups | 
					
						
							| 
									
										
										
										
											2024-05-22 20:12:59 -04:00
										 |  |  |     options_presets = tunic_option_presets | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TunicItem(Item): | 
					
						
							| 
									
										
										
										
											2024-01-17 19:56:34 -05:00
										 |  |  |     game: str = "TUNIC" | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TunicLocation(Location): | 
					
						
							| 
									
										
										
										
											2024-01-17 19:56:34 -05:00
										 |  |  |     game: str = "TUNIC" | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  | class SeedGroup(TypedDict): | 
					
						
							|  |  |  |     logic_rules: int  # logic rules value | 
					
						
							|  |  |  |     laurels_at_10_fairies: bool  # laurels location value | 
					
						
							|  |  |  |     fixed_shop: bool  # fixed shop value | 
					
						
							| 
									
										
										
										
											2024-06-03 04:44:37 -04:00
										 |  |  |     plando: TunicPlandoConnections  # consolidated of plando connections for the seed group | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | class TunicWorld(World): | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     Explore a land filled with lost legends, ancient powers, and ferocious monsters in TUNIC, an isometric action game | 
					
						
							|  |  |  |     about a small fox on a big adventure. Stranded on a mysterious beach, armed with only your own curiosity, you will | 
					
						
							|  |  |  |     confront colossal beasts, collect strange and powerful items, and unravel long-lost secrets. Be brave, tiny fox! | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2024-01-17 19:56:34 -05:00
										 |  |  |     game = "TUNIC" | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |     web = TunicWeb() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     options: TunicOptions | 
					
						
							|  |  |  |     options_dataclass = TunicOptions | 
					
						
							| 
									
										
										
										
											2024-07-23 03:04:24 -04:00
										 |  |  |     settings: ClassVar[TunicSettings] | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |     item_name_groups = item_name_groups | 
					
						
							|  |  |  |     location_name_groups = location_name_groups | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     item_name_to_id = item_name_to_id | 
					
						
							|  |  |  |     location_name_to_id = location_name_to_id | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ability_unlocks: Dict[str, int] | 
					
						
							|  |  |  |     slot_data_items: List[TunicItem] | 
					
						
							|  |  |  |     tunic_portal_pairs: Dict[str, str] | 
					
						
							|  |  |  |     er_portal_hints: Dict[int, str] | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  |     seed_groups: Dict[str, SeedGroup] = {} | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def generate_early(self) -> None: | 
					
						
							| 
									
										
										
										
											2024-06-01 06:34:41 -05:00
										 |  |  |         if self.options.plando_connections: | 
					
						
							|  |  |  |             for index, cxn in enumerate(self.options.plando_connections): | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  |                 # making shops second to simplify other things later | 
					
						
							|  |  |  |                 if cxn.entrance.startswith("Shop"): | 
					
						
							|  |  |  |                     replacement = PlandoConnection(cxn.exit, "Shop Portal", "both") | 
					
						
							| 
									
										
										
										
											2024-06-01 06:34:41 -05:00
										 |  |  |                     self.options.plando_connections.value.remove(cxn) | 
					
						
							|  |  |  |                     self.options.plando_connections.value.insert(index, replacement) | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  |                 elif cxn.exit.startswith("Shop"): | 
					
						
							|  |  |  |                     replacement = PlandoConnection(cxn.entrance, "Shop Portal", "both") | 
					
						
							| 
									
										
										
										
											2024-06-01 06:34:41 -05:00
										 |  |  |                     self.options.plando_connections.value.remove(cxn) | 
					
						
							|  |  |  |                     self.options.plando_connections.value.insert(index, replacement) | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 23:03:51 -05:00
										 |  |  |         # Universal tracker stuff, shouldn't do anything in standard gen | 
					
						
							|  |  |  |         if hasattr(self.multiworld, "re_gen_passthrough"): | 
					
						
							|  |  |  |             if "TUNIC" in self.multiworld.re_gen_passthrough: | 
					
						
							|  |  |  |                 passthrough = self.multiworld.re_gen_passthrough["TUNIC"] | 
					
						
							|  |  |  |                 self.options.start_with_sword.value = passthrough["start_with_sword"] | 
					
						
							|  |  |  |                 self.options.keys_behind_bosses.value = passthrough["keys_behind_bosses"] | 
					
						
							|  |  |  |                 self.options.sword_progression.value = passthrough["sword_progression"] | 
					
						
							|  |  |  |                 self.options.ability_shuffling.value = passthrough["ability_shuffling"] | 
					
						
							|  |  |  |                 self.options.logic_rules.value = passthrough["logic_rules"] | 
					
						
							|  |  |  |                 self.options.lanternless.value = passthrough["lanternless"] | 
					
						
							|  |  |  |                 self.options.maskless.value = passthrough["maskless"] | 
					
						
							|  |  |  |                 self.options.hexagon_quest.value = passthrough["hexagon_quest"] | 
					
						
							|  |  |  |                 self.options.entrance_rando.value = passthrough["entrance_rando"] | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |                 self.options.shuffle_ladders.value = passthrough["shuffle_ladders"] | 
					
						
							| 
									
										
										
										
											2024-06-03 04:44:37 -04:00
										 |  |  |                 self.options.fixed_shop.value = self.options.fixed_shop.option_false | 
					
						
							|  |  |  |                 self.options.laurels_location.value = self.options.laurels_location.option_anywhere | 
					
						
							| 
									
										
										
										
											2024-02-15 23:03:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def stage_generate_early(cls, multiworld: MultiWorld) -> None: | 
					
						
							|  |  |  |         tunic_worlds: Tuple[TunicWorld] = multiworld.get_game_worlds("TUNIC") | 
					
						
							|  |  |  |         for tunic in tunic_worlds: | 
					
						
							|  |  |  |             # if it's one of the options, then it isn't a custom seed group | 
					
						
							| 
									
										
										
										
											2024-06-03 04:44:37 -04:00
										 |  |  |             if tunic.options.entrance_rando.value in EntranceRando.options.values(): | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  |                 continue | 
					
						
							|  |  |  |             group = tunic.options.entrance_rando.value | 
					
						
							|  |  |  |             # if this is the first world in the group, set the rules equal to its rules | 
					
						
							|  |  |  |             if group not in cls.seed_groups: | 
					
						
							|  |  |  |                 cls.seed_groups[group] = SeedGroup(logic_rules=tunic.options.logic_rules.value, | 
					
						
							|  |  |  |                                                    laurels_at_10_fairies=tunic.options.laurels_location == 3, | 
					
						
							|  |  |  |                                                    fixed_shop=bool(tunic.options.fixed_shop), | 
					
						
							| 
									
										
										
										
											2024-07-27 17:17:59 -04:00
										 |  |  |                                                    plando=tunic.options.plando_connections) | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  |                 continue | 
					
						
							|  |  |  |                  | 
					
						
							|  |  |  |             # lower value is more restrictive | 
					
						
							|  |  |  |             if tunic.options.logic_rules.value < cls.seed_groups[group]["logic_rules"]: | 
					
						
							|  |  |  |                 cls.seed_groups[group]["logic_rules"] = tunic.options.logic_rules.value | 
					
						
							|  |  |  |             # laurels at 10 fairies changes logic for secret gathering place placement | 
					
						
							|  |  |  |             if tunic.options.laurels_location == 3: | 
					
						
							|  |  |  |                 cls.seed_groups[group]["laurels_at_10_fairies"] = True | 
					
						
							|  |  |  |             # fewer shops, one at windmill | 
					
						
							|  |  |  |             if tunic.options.fixed_shop: | 
					
						
							|  |  |  |                 cls.seed_groups[group]["fixed_shop"] = True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-27 17:17:59 -04:00
										 |  |  |             if tunic.options.plando_connections: | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  |                 # loop through the connections in the player's yaml | 
					
						
							| 
									
										
										
										
											2024-07-27 17:17:59 -04:00
										 |  |  |                 for cxn in tunic.options.plando_connections: | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  |                     new_cxn = True | 
					
						
							|  |  |  |                     for group_cxn in cls.seed_groups[group]["plando"]: | 
					
						
							|  |  |  |                         # if neither entrance nor exit match anything in the group, add to group | 
					
						
							|  |  |  |                         if ((cxn.entrance == group_cxn.entrance and cxn.exit == group_cxn.exit) | 
					
						
							|  |  |  |                                 or (cxn.exit == group_cxn.entrance and cxn.entrance == group_cxn.exit)): | 
					
						
							|  |  |  |                             new_cxn = False | 
					
						
							|  |  |  |                             break | 
					
						
							|  |  |  |                                     | 
					
						
							|  |  |  |                         # check if this pair is the same as a pair in the group already | 
					
						
							|  |  |  |                         is_mismatched = ( | 
					
						
							|  |  |  |                             cxn.entrance == group_cxn.entrance and cxn.exit != group_cxn.exit | 
					
						
							|  |  |  |                             or cxn.entrance == group_cxn.exit and cxn.exit != group_cxn.entrance | 
					
						
							|  |  |  |                             or cxn.exit == group_cxn.entrance and cxn.entrance != group_cxn.exit | 
					
						
							|  |  |  |                             or cxn.exit == group_cxn.exit and cxn.entrance != group_cxn.entrance | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                         if is_mismatched: | 
					
						
							|  |  |  |                             raise Exception(f"TUNIC: Conflict between seed group {group}'s plando " | 
					
						
							|  |  |  |                                             f"connection {group_cxn.entrance} <-> {group_cxn.exit} and " | 
					
						
							| 
									
										
										
										
											2024-08-13 20:35:08 -04:00
										 |  |  |                                             f"{tunic.player_name}'s plando connection {cxn.entrance} <-> {cxn.exit}") | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  |                     if new_cxn: | 
					
						
							| 
									
										
										
										
											2024-06-03 04:44:37 -04:00
										 |  |  |                         cls.seed_groups[group]["plando"].value.append(cxn) | 
					
						
							| 
									
										
										
										
											2024-05-03 01:21:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-24 08:37:18 -04:00
										 |  |  |     def create_item(self, name: str, classification: ItemClassification = None) -> TunicItem: | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |         item_data = item_table[name] | 
					
						
							| 
									
										
										
										
											2024-07-24 08:37:18 -04:00
										 |  |  |         return TunicItem(name, classification or item_data.classification, self.item_name_to_id[name], self.player) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def create_items(self) -> None: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         tunic_items: List[TunicItem] = [] | 
					
						
							|  |  |  |         self.slot_data_items = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         items_to_create: Dict[str, int] = {item: data.quantity_in_item_pool for item, data in item_table.items()} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for money_fool in fool_tiers[self.options.fool_traps]: | 
					
						
							|  |  |  |             items_to_create["Fool Trap"] += items_to_create[money_fool] | 
					
						
							|  |  |  |             items_to_create[money_fool] = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-23 13:41:59 -05:00
										 |  |  |         if self.options.start_with_sword: | 
					
						
							|  |  |  |             self.multiworld.push_precollected(self.create_item("Sword")) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-29 19:17:00 -04:00
										 |  |  |         if self.options.sword_progression: | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             items_to_create["Stick"] = 0 | 
					
						
							|  |  |  |             items_to_create["Sword"] = 0 | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             items_to_create["Sword Upgrade"] = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if self.options.laurels_location: | 
					
						
							|  |  |  |             laurels = self.create_item("Hero's Laurels") | 
					
						
							|  |  |  |             if self.options.laurels_location == "6_coins": | 
					
						
							| 
									
										
										
										
											2024-08-13 20:35:08 -04:00
										 |  |  |                 self.get_location("Coins in the Well - 6 Coins").place_locked_item(laurels) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             elif self.options.laurels_location == "10_coins": | 
					
						
							| 
									
										
										
										
											2024-08-13 20:35:08 -04:00
										 |  |  |                 self.get_location("Coins in the Well - 10 Coins").place_locked_item(laurels) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             elif self.options.laurels_location == "10_fairies": | 
					
						
							| 
									
										
										
										
											2024-08-13 20:35:08 -04:00
										 |  |  |                 self.get_location("Secret Gathering Place - 10 Fairy Reward").place_locked_item(laurels) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             items_to_create["Hero's Laurels"] = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-29 19:17:00 -04:00
										 |  |  |         if self.options.keys_behind_bosses: | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             for rgb_hexagon, location in hexagon_locations.items(): | 
					
						
							| 
									
										
										
										
											2024-06-29 19:17:00 -04:00
										 |  |  |                 hex_item = self.create_item(gold_hexagon if self.options.hexagon_quest else rgb_hexagon) | 
					
						
							| 
									
										
										
										
											2024-08-13 20:35:08 -04:00
										 |  |  |                 self.get_location(location).place_locked_item(hex_item) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |                 items_to_create[rgb_hexagon] = 0 | 
					
						
							|  |  |  |             items_to_create[gold_hexagon] -= 3 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |         # Filler items in the item pool | 
					
						
							|  |  |  |         available_filler: List[str] = [filler for filler in items_to_create if items_to_create[filler] > 0 and | 
					
						
							|  |  |  |                                        item_table[filler].classification == ItemClassification.filler] | 
					
						
							| 
									
										
										
										
											2024-04-13 18:20:52 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |         # Remove filler to make room for other items | 
					
						
							| 
									
										
										
										
											2024-04-13 20:06:06 -04:00
										 |  |  |         def remove_filler(amount: int) -> None: | 
					
						
							| 
									
										
										
										
											2024-07-05 16:50:12 -04:00
										 |  |  |             for _ in range(amount): | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |                 if not available_filler: | 
					
						
							|  |  |  |                     fill = "Fool Trap" | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     fill = self.random.choice(available_filler) | 
					
						
							|  |  |  |                 if items_to_create[fill] == 0: | 
					
						
							|  |  |  |                     raise Exception("No filler items left to accommodate options selected. Turn down fool trap amount.") | 
					
						
							|  |  |  |                 items_to_create[fill] -= 1 | 
					
						
							|  |  |  |                 if items_to_create[fill] == 0: | 
					
						
							|  |  |  |                     available_filler.remove(fill) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if self.options.shuffle_ladders: | 
					
						
							|  |  |  |             ladder_count = 0 | 
					
						
							|  |  |  |             for item_name, item_data in item_table.items(): | 
					
						
							| 
									
										
										
										
											2024-05-02 04:02:59 -04:00
										 |  |  |                 if item_data.item_group == "Ladders": | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |                     items_to_create[item_name] = 1 | 
					
						
							|  |  |  |                     ladder_count += 1 | 
					
						
							|  |  |  |             remove_filler(ladder_count) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-29 19:17:00 -04:00
										 |  |  |         if self.options.hexagon_quest: | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             # Calculate number of hexagons in item pool | 
					
						
							|  |  |  |             hexagon_goal = self.options.hexagon_goal | 
					
						
							|  |  |  |             extra_hexagons = self.options.extra_hexagon_percentage | 
					
						
							|  |  |  |             items_to_create[gold_hexagon] += int((Decimal(100 + extra_hexagons) / 100 * hexagon_goal).to_integral_value(rounding=ROUND_HALF_UP)) | 
					
						
							| 
									
										
										
										
											2024-04-13 18:20:52 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             # Replace pages and normal hexagons with filler | 
					
						
							|  |  |  |             for replaced_item in list(filter(lambda item: "Pages" in item or item in hexagon_locations, items_to_create)): | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |                 filler_name = self.get_filler_item_name() | 
					
						
							|  |  |  |                 items_to_create[filler_name] += items_to_create[replaced_item] | 
					
						
							|  |  |  |                 if items_to_create[filler_name] >= 1 and filler_name not in available_filler: | 
					
						
							|  |  |  |                     available_filler.append(filler_name) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |                 items_to_create[replaced_item] = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |             remove_filler(items_to_create[gold_hexagon]) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-29 19:17:00 -04:00
										 |  |  |             for hero_relic in item_name_groups["Hero Relics"]: | 
					
						
							| 
									
										
										
										
											2024-07-24 08:37:18 -04:00
										 |  |  |                 tunic_items.append(self.create_item(hero_relic, ItemClassification.useful)) | 
					
						
							| 
									
										
										
										
											2024-06-29 19:17:00 -04:00
										 |  |  |                 items_to_create[hero_relic] = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not self.options.ability_shuffling: | 
					
						
							|  |  |  |             for page in item_name_groups["Abilities"]: | 
					
						
							|  |  |  |                 if items_to_create[page] > 0: | 
					
						
							| 
									
										
										
										
											2024-07-24 08:37:18 -04:00
										 |  |  |                     tunic_items.append(self.create_item(page, ItemClassification.useful)) | 
					
						
							| 
									
										
										
										
											2024-06-29 19:17:00 -04:00
										 |  |  |                     items_to_create[page] = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |         if self.options.maskless: | 
					
						
							| 
									
										
										
										
											2024-07-24 08:37:18 -04:00
										 |  |  |             tunic_items.append(self.create_item("Scavenger Mask", ItemClassification.useful)) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             items_to_create["Scavenger Mask"] = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if self.options.lanternless: | 
					
						
							| 
									
										
										
										
											2024-07-24 08:37:18 -04:00
										 |  |  |             tunic_items.append(self.create_item("Lantern", ItemClassification.useful)) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             items_to_create["Lantern"] = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for item, quantity in items_to_create.items(): | 
					
						
							| 
									
										
										
										
											2024-07-05 16:50:12 -04:00
										 |  |  |             for _ in range(quantity): | 
					
						
							| 
									
										
										
										
											2024-07-24 08:37:18 -04:00
										 |  |  |                 tunic_items.append(self.create_item(item)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for tunic_item in tunic_items: | 
					
						
							|  |  |  |             if tunic_item.name in slot_data_item_names: | 
					
						
							|  |  |  |                 self.slot_data_items.append(tunic_item) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.multiworld.itempool += tunic_items | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def create_regions(self) -> None: | 
					
						
							|  |  |  |         self.tunic_portal_pairs = {} | 
					
						
							|  |  |  |         self.er_portal_hints = {} | 
					
						
							|  |  |  |         self.ability_unlocks = randomize_ability_unlocks(self.random, self.options) | 
					
						
							| 
									
										
										
										
											2024-04-13 18:20:52 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-15 23:03:51 -05:00
										 |  |  |         # stuff for universal tracker support, can be ignored for standard gen | 
					
						
							|  |  |  |         if hasattr(self.multiworld, "re_gen_passthrough"): | 
					
						
							|  |  |  |             if "TUNIC" in self.multiworld.re_gen_passthrough: | 
					
						
							|  |  |  |                 passthrough = self.multiworld.re_gen_passthrough["TUNIC"] | 
					
						
							|  |  |  |                 self.ability_unlocks["Pages 24-25 (Prayer)"] = passthrough["Hexagon Quest Prayer"] | 
					
						
							|  |  |  |                 self.ability_unlocks["Pages 42-43 (Holy Cross)"] = passthrough["Hexagon Quest Holy Cross"] | 
					
						
							| 
									
										
										
										
											2024-02-16 17:25:20 -05:00
										 |  |  |                 self.ability_unlocks["Pages 52-53 (Icebolt)"] = passthrough["Hexagon Quest Icebolt"] | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |         # ladder rando uses ER with vanilla connections, so that we're not managing more rules files | 
					
						
							|  |  |  |         if self.options.entrance_rando or self.options.shuffle_ladders: | 
					
						
							|  |  |  |             portal_pairs = create_er_regions(self) | 
					
						
							|  |  |  |             if self.options.entrance_rando: | 
					
						
							|  |  |  |                 # these get interpreted by the game to tell it which entrances to connect | 
					
						
							|  |  |  |                 for portal1, portal2 in portal_pairs.items(): | 
					
						
							|  |  |  |                     self.tunic_portal_pairs[portal1.scene_destination()] = portal2.scene_destination() | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |             # for non-ER, non-ladders | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             for region_name in tunic_regions: | 
					
						
							|  |  |  |                 region = Region(region_name, self.player, self.multiworld) | 
					
						
							|  |  |  |                 self.multiworld.regions.append(region) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for region_name, exits in tunic_regions.items(): | 
					
						
							| 
									
										
										
										
											2024-08-13 20:35:08 -04:00
										 |  |  |                 region = self.get_region(region_name) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |                 region.add_exits(exits) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for location_name, location_id in self.location_name_to_id.items(): | 
					
						
							| 
									
										
										
										
											2024-08-13 20:35:08 -04:00
										 |  |  |                 region = self.get_region(location_table[location_name].region) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |                 location = TunicLocation(self.player, location_name, location_id, region) | 
					
						
							|  |  |  |                 region.locations.append(location) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-13 20:35:08 -04:00
										 |  |  |             victory_region = self.get_region("Spirit Arena") | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             victory_location = TunicLocation(self.player, "The Heir", None, victory_region) | 
					
						
							|  |  |  |             victory_location.place_locked_item(TunicItem("Victory", ItemClassification.progression, None, self.player)) | 
					
						
							|  |  |  |             self.multiworld.completion_condition[self.player] = lambda state: state.has("Victory", self.player) | 
					
						
							|  |  |  |             victory_region.locations.append(victory_location) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_rules(self) -> None: | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |         if self.options.entrance_rando or self.options.shuffle_ladders: | 
					
						
							| 
									
										
										
										
											2024-07-05 16:50:12 -04:00
										 |  |  |             set_er_location_rules(self) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2024-07-05 16:50:12 -04:00
										 |  |  |             set_region_rules(self) | 
					
						
							|  |  |  |             set_location_rules(self) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def get_filler_item_name(self) -> str: | 
					
						
							|  |  |  |         return self.random.choice(filler_items) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-13 20:06:06 -04:00
										 |  |  |     def extend_hint_information(self, hint_data: Dict[int, Dict[int, str]]) -> None: | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |         if self.options.entrance_rando: | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |             hint_data.update({self.player: {}}) | 
					
						
							|  |  |  |             # all state seems to have efficient paths | 
					
						
							|  |  |  |             all_state = self.multiworld.get_all_state(True) | 
					
						
							|  |  |  |             all_state.update_reachable_regions(self.player) | 
					
						
							|  |  |  |             paths = all_state.path | 
					
						
							|  |  |  |             portal_names = [portal.name for portal in portal_mapping] | 
					
						
							|  |  |  |             for location in self.multiworld.get_locations(self.player): | 
					
						
							|  |  |  |                 # skipping event locations | 
					
						
							|  |  |  |                 if not location.address: | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 path_to_loc = [] | 
					
						
							|  |  |  |                 previous_name = "placeholder" | 
					
						
							| 
									
										
										
										
											2024-04-13 18:20:52 -04:00
										 |  |  |                 try: | 
					
						
							|  |  |  |                     name, connection = paths[location.parent_region] | 
					
						
							|  |  |  |                 except KeyError: | 
					
						
							|  |  |  |                     # logic bug, proceed with warning since it takes a long time to update AP | 
					
						
							| 
									
										
										
										
											2024-08-13 20:35:08 -04:00
										 |  |  |                     warning(f"{location.name} is not logically accessible for {self.player_name}. " | 
					
						
							| 
									
										
										
										
											2024-08-29 03:42:46 -04:00
										 |  |  |                             "Creating entrance hint Inaccessible. Please report this to the TUNIC rando devs. " | 
					
						
							|  |  |  |                             "If you are using Plando Items (excluding early locations), then this is likely the cause.") | 
					
						
							| 
									
										
										
										
											2024-04-13 18:20:52 -04:00
										 |  |  |                     hint_text = "Inaccessible" | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     while connection != ("Menu", None): | 
					
						
							|  |  |  |                         name, connection = connection | 
					
						
							|  |  |  |                         # for LS entrances, we just want to give the portal name | 
					
						
							|  |  |  |                         if "(LS)" in name: | 
					
						
							| 
									
										
										
										
											2024-05-02 04:02:59 -04:00
										 |  |  |                             name = name.split(" (LS) ", 1)[0] | 
					
						
							| 
									
										
										
										
											2024-04-13 18:20:52 -04:00
										 |  |  |                         # was getting some cases like Library Grave -> Library Grave -> other place | 
					
						
							|  |  |  |                         if name in portal_names and name != previous_name: | 
					
						
							|  |  |  |                             previous_name = name | 
					
						
							|  |  |  |                             path_to_loc.append(name) | 
					
						
							|  |  |  |                     hint_text = " -> ".join(reversed(path_to_loc)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |                 if hint_text: | 
					
						
							|  |  |  |                     hint_data[self.player][location.address] = hint_text | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def fill_slot_data(self) -> Dict[str, Any]: | 
					
						
							|  |  |  |         slot_data: Dict[str, Any] = { | 
					
						
							|  |  |  |             "seed": self.random.randint(0, 2147483647), | 
					
						
							|  |  |  |             "start_with_sword": self.options.start_with_sword.value, | 
					
						
							|  |  |  |             "keys_behind_bosses": self.options.keys_behind_bosses.value, | 
					
						
							|  |  |  |             "sword_progression": self.options.sword_progression.value, | 
					
						
							|  |  |  |             "ability_shuffling": self.options.ability_shuffling.value, | 
					
						
							|  |  |  |             "hexagon_quest": self.options.hexagon_quest.value, | 
					
						
							|  |  |  |             "fool_traps": self.options.fool_traps.value, | 
					
						
							| 
									
										
										
										
											2024-02-15 23:03:51 -05:00
										 |  |  |             "logic_rules": self.options.logic_rules.value, | 
					
						
							|  |  |  |             "lanternless": self.options.lanternless.value, | 
					
						
							|  |  |  |             "maskless": self.options.maskless.value, | 
					
						
							| 
									
										
										
										
											2024-03-21 11:50:07 -04:00
										 |  |  |             "entrance_rando": int(bool(self.options.entrance_rando.value)), | 
					
						
							|  |  |  |             "shuffle_ladders": self.options.shuffle_ladders.value, | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             "Hexagon Quest Prayer": self.ability_unlocks["Pages 24-25 (Prayer)"], | 
					
						
							|  |  |  |             "Hexagon Quest Holy Cross": self.ability_unlocks["Pages 42-43 (Holy Cross)"], | 
					
						
							| 
									
										
										
										
											2024-02-16 17:25:20 -05:00
										 |  |  |             "Hexagon Quest Icebolt": self.ability_unlocks["Pages 52-53 (Icebolt)"], | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |             "Hexagon Quest Goal": self.options.hexagon_goal.value, | 
					
						
							| 
									
										
										
										
											2024-07-23 03:04:24 -04:00
										 |  |  |             "Entrance Rando": self.tunic_portal_pairs, | 
					
						
							|  |  |  |             "disable_local_spoiler": int(self.settings.disable_local_spoiler or self.multiworld.is_race), | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for tunic_item in filter(lambda item: item.location is not None and item.code is not None, self.slot_data_items): | 
					
						
							|  |  |  |             if tunic_item.name not in slot_data: | 
					
						
							|  |  |  |                 slot_data[tunic_item.name] = [] | 
					
						
							|  |  |  |             if tunic_item.name == gold_hexagon and len(slot_data[gold_hexagon]) >= 6: | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             slot_data[tunic_item.name].extend([tunic_item.location.name, tunic_item.location.player]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for start_item in self.options.start_inventory_from_pool: | 
					
						
							|  |  |  |             if start_item in slot_data_item_names: | 
					
						
							|  |  |  |                 if start_item not in slot_data: | 
					
						
							|  |  |  |                     slot_data[start_item] = [] | 
					
						
							| 
									
										
										
										
											2024-07-05 16:50:12 -04:00
										 |  |  |                 for _ in range(self.options.start_inventory_from_pool[start_item]): | 
					
						
							| 
									
										
										
										
											2024-01-12 14:32:15 -05:00
										 |  |  |                     slot_data[start_item].extend(["Your Pocket", self.player]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for plando_item in self.multiworld.plando_items[self.player]: | 
					
						
							|  |  |  |             if plando_item["from_pool"]: | 
					
						
							|  |  |  |                 items_to_find = set() | 
					
						
							|  |  |  |                 for item_type in [key for key in ["item", "items"] if key in plando_item]: | 
					
						
							|  |  |  |                     for item in plando_item[item_type]: | 
					
						
							|  |  |  |                         items_to_find.add(item) | 
					
						
							|  |  |  |                 for item in items_to_find: | 
					
						
							|  |  |  |                     if item in slot_data_item_names: | 
					
						
							|  |  |  |                         slot_data[item] = [] | 
					
						
							|  |  |  |                         for item_location in self.multiworld.find_item_locations(item, self.player): | 
					
						
							|  |  |  |                             slot_data[item].extend([item_location.name, item_location.player]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return slot_data | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # for the universal tracker, doesn't get called in standard gen | 
					
						
							| 
									
										
										
										
											2024-08-11 18:24:30 -04:00
										 |  |  |     # docs: https://github.com/FarisTheAncient/Archipelago/blob/tracker/worlds/tracker/docs/re-gen-passthrough.md | 
					
						
							| 
									
										
										
										
											2024-02-15 23:03:51 -05:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def interpret_slot_data(slot_data: Dict[str, Any]) -> Dict[str, Any]: | 
					
						
							|  |  |  |         # returning slot_data so it regens, giving it back in multiworld.re_gen_passthrough | 
					
						
							| 
									
										
										
										
											2024-08-11 18:24:30 -04:00
										 |  |  |         # we are using re_gen_passthrough over modifying the world here due to complexities with ER | 
					
						
							| 
									
										
										
										
											2024-02-15 23:03:51 -05:00
										 |  |  |         return slot_data |