mirror of
				https://github.com/MarioSpore/Grinch-AP.git
				synced 2025-10-21 20:21:32 -06:00 
			
		
		
		
	 a6e1e14fee
			
		
	
	a6e1e14fee
	
	
	
		
			
			* OoT: ER improvements Include dungeon rewards in itempool to allow for ER improvement Better validate_world function by checking for multi-entrance incompatibility more efficiently Fix some generation failures by ensuring all entrances placed with logic Introduce bias to some interior entrance placement to improve generation rate * OoT: fix overworld ER spoiler information * OoT: rewrite dungeon item placement algorithm in particular, no longer assumes that exactly the number of vanilla keys is present, which lets it place more or fewer items. * OoT: auto-send more locations Now should autosend cows, DMT/DMC great fairies, medigoron, and bombchu salesman This should be every check autosending. these ones are super weird for some reason and didn't get fixed with the others * OoT: add items forced local by settings to AP's local_items * OoT: fast-fill shop junk items * OoT: ensure that Kokiri Shop is always reachable immediately in closed forest hence Deku Shield can be bought to leave the forest * OoT: randomize internal connect name Connect name is now a random 16-character string. This should prevent any issues with connecting to a room with the wrong ROM with probability almost 1. * OoT: introduce TrackRandomRange for trials hint and mq dungeon maps * OoT: enable proper itemlinking of songs and dungeon items, with restricted placements according to player settings * OoT: barren hint oversight fix * OoT: allow NL + ER to roll properly * OoT: 3.8 compatibility set and list builtins don't have proper typing support until 3.9, apparently * OoT: remove Gerudo Membership Card location from the pool if fortress open and card not randomized another long-standing bug squished * OoT: exclude locations in the itemlink song fill if they aren't also priority * OoT: prevent data bleed when client isn't closed between different game connections I don't understand why people keep doing this * OoT: linter appeasement it was a real error though * fixing merge conflicts is hard * oot merge update #2 * OoT: removed accidentally duplicated code
		
			
				
	
	
		
			227 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			227 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from collections import deque
 | |
| import logging
 | |
| 
 | |
| from .SaveContext import SaveContext
 | |
| from .Regions import TimeOfDay
 | |
| from .Items import oot_is_item_of_type
 | |
| 
 | |
| from BaseClasses import CollectionState
 | |
| from worlds.generic.Rules import set_rule, add_rule, add_item_rule, forbid_item
 | |
| from ..AutoWorld import LogicMixin
 | |
| 
 | |
| 
 | |
| class OOTLogic(LogicMixin):
 | |
| 
 | |
|     def _oot_has_stones(self, count, player): 
 | |
|         return self.has_group("stones", player, count)
 | |
| 
 | |
|     def _oot_has_medallions(self, count, player): 
 | |
|         return self.has_group("medallions", player, count)
 | |
| 
 | |
|     def _oot_has_dungeon_rewards(self, count, player): 
 | |
|         return self.has_group("rewards", player, count)
 | |
| 
 | |
|     def _oot_has_bottle(self, player): 
 | |
|         return self.has_group("bottles", player)
 | |
| 
 | |
|     # Used for fall damage and other situations where damage is unavoidable
 | |
|     def _oot_can_live_dmg(self, player, hearts):
 | |
|         mult = self.multiworld.worlds[player].damage_multiplier
 | |
|         if hearts*4 >= 3:
 | |
|             return mult != 'ohko' and mult != 'quadruple'
 | |
|         else:
 | |
|             return mult != 'ohko'
 | |
| 
 | |
|     # This function operates by assuming different behavior based on the "level of recursion", handled manually. 
 | |
|     # If it's called while self.age[player] is None, then it will set the age variable and then attempt to reach the region. 
 | |
|     # If self.age[player] is not None, then it will compare it to the 'age' parameter, and return True iff they are equal. 
 | |
|     #   This lets us fake the OOT accessibility check that cares about age. Unfortunately it's still tied to the ground region. 
 | |
|     def _oot_reach_as_age(self, regionname, age, player): 
 | |
|         if self.age[player] is None: 
 | |
|             self.age[player] = age
 | |
|             can_reach = self.multiworld.get_region(regionname, player).can_reach(self)
 | |
|             self.age[player] = None
 | |
|             return can_reach
 | |
|         return self.age[player] == age
 | |
| 
 | |
|     def _oot_reach_at_time(self, regionname, tod, already_checked, player):
 | |
|         name_map = {
 | |
|             TimeOfDay.DAY: self.day_reachable_regions[player],
 | |
|             TimeOfDay.DAMPE: self.dampe_reachable_regions[player],
 | |
|             TimeOfDay.ALL: self.day_reachable_regions[player].intersection(self.dampe_reachable_regions[player])
 | |
|         }
 | |
|         if regionname in name_map[tod]:
 | |
|             return True
 | |
|         region = self.multiworld.get_region(regionname, player)
 | |
|         if region.provides_time == TimeOfDay.ALL or regionname == 'Root':
 | |
|             self.day_reachable_regions[player].add(regionname)
 | |
|             self.dampe_reachable_regions[player].add(regionname)
 | |
|             return True
 | |
|         if region.provides_time == TimeOfDay.DAMPE:
 | |
|             self.dampe_reachable_regions[player].add(regionname)
 | |
|             return tod == TimeOfDay.DAMPE
 | |
|         for entrance in region.entrances:
 | |
|             if entrance.parent_region.name in already_checked:
 | |
|                 continue
 | |
|             if self._oot_reach_at_time(entrance.parent_region.name, tod, already_checked + [regionname], player):
 | |
|                 if tod == TimeOfDay.DAY:
 | |
|                     self.day_reachable_regions[player].add(regionname)
 | |
|                 elif tod == TimeOfDay.DAMPE:
 | |
|                     self.dampe_reachable_regions[player].add(regionname)
 | |
|                 elif tod == TimeOfDay.ALL:
 | |
|                     self.day_reachable_regions[player].add(regionname)
 | |
|                     self.dampe_reachable_regions[player].add(regionname)
 | |
|                 return True
 | |
|         return False
 | |
| 
 | |
|     # Store the age before calling this!
 | |
|     def _oot_update_age_reachable_regions(self, player): 
 | |
|         self.stale[player] = False
 | |
|         for age in ['child', 'adult']: 
 | |
|             self.age[player] = age
 | |
|             rrp = getattr(self, f'{age}_reachable_regions')[player]
 | |
|             bc = getattr(self, f'{age}_blocked_connections')[player]
 | |
|             queue = deque(getattr(self, f'{age}_blocked_connections')[player])
 | |
|             start = self.multiworld.get_region('Menu', player)
 | |
| 
 | |
|             # init on first call - this can't be done on construction since the regions don't exist yet
 | |
|             if not start in rrp:
 | |
|                 rrp.add(start)
 | |
|                 bc.update(start.exits)
 | |
|                 queue.extend(start.exits)
 | |
| 
 | |
|             # run BFS on all connections, and keep track of those blocked by missing items
 | |
|             while queue:
 | |
|                 connection = queue.popleft()
 | |
|                 new_region = connection.connected_region
 | |
|                 if new_region is None: 
 | |
|                     continue
 | |
|                 if new_region in rrp:
 | |
|                     bc.remove(connection)
 | |
|                 elif connection.can_reach(self):
 | |
|                     rrp.add(new_region)
 | |
|                     bc.remove(connection)
 | |
|                     bc.update(new_region.exits)
 | |
|                     queue.extend(new_region.exits)
 | |
|                     self.path[new_region] = (new_region.name, self.path.get(connection, None))
 | |
| 
 | |
| 
 | |
| # Sets extra rules on various specific locations not handled by the rule parser.
 | |
| def set_rules(ootworld):
 | |
|     logger = logging.getLogger('')
 | |
| 
 | |
|     world = ootworld.multiworld
 | |
|     player = ootworld.player
 | |
| 
 | |
|     if ootworld.logic_rules != 'no_logic': 
 | |
|         if ootworld.triforce_hunt: 
 | |
|             world.completion_condition[player] = lambda state: state.has('Triforce Piece', player, ootworld.triforce_goal)
 | |
|         else: 
 | |
|             world.completion_condition[player] = lambda state: state.has('Triforce', player)
 | |
| 
 | |
|     # ganon can only carry triforce
 | |
|     world.get_location('Ganon', player).item_rule = lambda item: item.name == 'Triforce'
 | |
| 
 | |
|     # is_child = ootworld.parser.parse_rule('is_child')
 | |
|     guarantee_hint = ootworld.parser.parse_rule('guarantee_hint')
 | |
| 
 | |
|     for location in filter(lambda location: location.name in ootworld.shop_prices or 'Deku Scrub' in location.name, ootworld.get_locations()):
 | |
|         if location.type == 'Shop':
 | |
|             location.price = ootworld.shop_prices[location.name]
 | |
|         add_rule(location, create_shop_rule(location, ootworld.parser))
 | |
| 
 | |
|     if ootworld.dungeon_mq['Forest Temple'] and ootworld.shuffle_bosskeys == 'dungeon' and ootworld.shuffle_smallkeys == 'dungeon' and ootworld.tokensanity == 'off':
 | |
|         # First room chest needs to be a small key. Make sure the boss key isn't placed here.
 | |
|         location = world.get_location('Forest Temple MQ First Room Chest', player)
 | |
|         forbid_item(location, 'Boss Key (Forest Temple)', ootworld.player)
 | |
| 
 | |
|     if ootworld.shuffle_song_items == 'song' and not ootworld.songs_as_items:
 | |
|         # Sheik in Ice Cavern is the only song location in a dungeon; need to ensure that it cannot be anything else.
 | |
|         # This is required if map/compass included, or any_dungeon shuffle.
 | |
|         location = world.get_location('Sheik in Ice Cavern', player)
 | |
|         add_item_rule(location, lambda item: item.player == player and oot_is_item_of_type(item, 'Song'))
 | |
| 
 | |
|     if ootworld.skip_child_zelda:
 | |
|         # If skip child zelda is on, the item at Song from Impa must be giveable by the save context. 
 | |
|         location = world.get_location('Song from Impa', player)
 | |
|         add_item_rule(location, lambda item: item.name in SaveContext.giveable_items)
 | |
| 
 | |
|     for name in ootworld.always_hints:
 | |
|         add_rule(world.get_location(name, player), guarantee_hint)
 | |
| 
 | |
|     # TODO: re-add hints once they are working
 | |
|     # if location.type == 'HintStone' and ootworld.hints == 'mask':
 | |
|     #     location.add_rule(is_child)
 | |
| 
 | |
| 
 | |
| def create_shop_rule(location, parser):
 | |
|     def required_wallets(price):
 | |
|         if price > 500:
 | |
|             return 3
 | |
|         if price > 200:
 | |
|             return 2
 | |
|         if price > 99:
 | |
|             return 1
 | |
|         return 0
 | |
|     return parser.parse_rule('(Progressive_Wallet, %d)' % required_wallets(location.price))
 | |
| 
 | |
| 
 | |
| def limit_to_itemset(location, itemset):
 | |
|     old_rule = location.item_rule
 | |
|     location.item_rule = lambda item: item.name in itemset and old_rule(item)
 | |
| 
 | |
| 
 | |
| # This function should be run once after the shop items are placed in the world.
 | |
| # It should be run before other items are placed in the world so that logic has
 | |
| # the correct checks for them. This is safe to do since every shop is still
 | |
| # accessible when all items are obtained and every shop item is not.
 | |
| # This function should also be called when a world is copied if the original world
 | |
| # had called this function because the world.copy does not copy the rules
 | |
| def set_shop_rules(ootworld):
 | |
|     found_bombchus = ootworld.parser.parse_rule('found_bombchus')
 | |
|     wallet = ootworld.parser.parse_rule('Progressive_Wallet')
 | |
|     wallet2 = ootworld.parser.parse_rule('(Progressive_Wallet, 2)')
 | |
| 
 | |
|     for location in filter(lambda location: location.item and oot_is_item_of_type(location.item, 'Shop'), ootworld.get_locations()):
 | |
|         # Add wallet requirements
 | |
|         if location.item.name in ['Buy Arrows (50)', 'Buy Fish', 'Buy Goron Tunic', 'Buy Bombchu (20)', 'Buy Bombs (30)']:
 | |
|             add_rule(location, wallet)
 | |
|         elif location.item.name in ['Buy Zora Tunic', 'Buy Blue Fire']:
 | |
|             add_rule(location, wallet2)
 | |
| 
 | |
|         # Add adult only checks
 | |
|         if location.item.name in ['Buy Goron Tunic', 'Buy Zora Tunic']:
 | |
|             add_rule(location, ootworld.parser.parse_rule('is_adult', location))
 | |
| 
 | |
|         # Add item prerequisite checks
 | |
|         if location.item.name in ['Buy Blue Fire',
 | |
|                                   'Buy Blue Potion',
 | |
|                                   'Buy Bottle Bug',
 | |
|                                   'Buy Fish',
 | |
|                                   'Buy Green Potion',
 | |
|                                   'Buy Poe',
 | |
|                                   'Buy Red Potion [30]',
 | |
|                                   'Buy Red Potion [40]',
 | |
|                                   'Buy Red Potion [50]',
 | |
|                                   'Buy Fairy\'s Spirit']:
 | |
|             add_rule(location, lambda state: CollectionState._oot_has_bottle(state, ootworld.player))
 | |
|         if location.item.name in ['Buy Bombchu (10)', 'Buy Bombchu (20)', 'Buy Bombchu (5)']:
 | |
|             add_rule(location, found_bombchus)
 | |
| 
 | |
| 
 | |
| # This function should be ran once after setting up entrances and before placing items
 | |
| # The goal is to automatically set item rules based on age requirements in case entrances were shuffled
 | |
| def set_entrances_based_rules(ootworld):
 | |
| 
 | |
|     if ootworld.multiworld.accessibility == 'beatable': 
 | |
|         return
 | |
| 
 | |
|     all_state = ootworld.multiworld.get_all_state(False)
 | |
| 
 | |
|     for location in filter(lambda location: location.type == 'Shop', ootworld.get_locations()):
 | |
|         # If a shop is not reachable as adult, it can't have Goron Tunic or Zora Tunic as child can't buy these
 | |
|         if not all_state._oot_reach_as_age(location.parent_region.name, 'adult', ootworld.player):
 | |
|             forbid_item(location, 'Buy Goron Tunic', ootworld.player)
 | |
|             forbid_item(location, 'Buy Zora Tunic', ootworld.player)
 | |
| 
 |