mirror of
				https://github.com/MarioSpore/Grinch-AP.git
				synced 2025-10-21 20:21:32 -06:00 
			
		
		
		
	
		
			
				
	
	
		
			591 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			591 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import copy
 | |
| import logging
 | |
| import json
 | |
| from collections import OrderedDict
 | |
| 
 | |
| 
 | |
| class World(object):
 | |
| 
 | |
|     def __init__(self, shuffle, logic, mode, difficulty, goal, algorithm, place_dungeon_items, check_beatable_only, shuffle_ganon):
 | |
|         self.shuffle = shuffle
 | |
|         self.logic = logic
 | |
|         self.mode = mode
 | |
|         self.difficulty = difficulty
 | |
|         self.goal = goal
 | |
|         self.algorithm = algorithm
 | |
|         self.regions = []
 | |
|         self.itempool = []
 | |
|         self.seed = None
 | |
|         self.state = CollectionState(self)
 | |
|         self.required_medallions = ['Ether', 'Quake']
 | |
|         self._cached_locations = None
 | |
|         self._entrance_cache = {}
 | |
|         self._region_cache = {}
 | |
|         self._entrance_cache = {}
 | |
|         self._location_cache = {}
 | |
|         self._item_cache = {}
 | |
|         self.required_locations = []
 | |
|         self.place_dungeon_items = place_dungeon_items  # configurable in future
 | |
|         self.shuffle_bonk_prizes = False
 | |
|         self.swamp_patch_required = False
 | |
|         self.sewer_light_cone = mode == 'standard'
 | |
|         self.light_world_light_cone = False
 | |
|         self.dark_world_light_cone = False
 | |
|         self.treasure_hunt_count = 0
 | |
|         self.treasure_hunt_icon = 'Power Star'
 | |
|         self.clock_mode = 'off'
 | |
|         self.aga_randomness = False
 | |
|         self.lock_aga_door_in_escape = False
 | |
|         self.fix_trock_doors = self.shuffle != 'vanilla'
 | |
|         self.save_and_quite_from_boss = False
 | |
|         self.check_beatable_only = check_beatable_only
 | |
|         self.fix_skullwoods_exit = self.shuffle not in ['vanilla', 'simple', 'restricted', 'dungeonssimple']
 | |
|         self.fix_palaceofdarkness_exit = self.shuffle not in ['vanilla', 'simple', 'restricted', 'dungeonssimple']
 | |
|         self.fix_trock_exit = self.shuffle not in ['vanilla', 'simple', 'restricted', 'dungeonssimple']
 | |
|         self.shuffle_ganon = shuffle_ganon
 | |
|         self.fix_gtower_exit = self.shuffle_ganon
 | |
|         self.can_access_trock_eyebridge = None
 | |
|         self.spoiler = Spoiler(self)
 | |
| 
 | |
|     def get_region(self, regionname):
 | |
|         if isinstance(regionname, Region):
 | |
|             return regionname
 | |
|         try:
 | |
|             return self._region_cache[regionname]
 | |
|         except KeyError:
 | |
|             for region in self.regions:
 | |
|                 if region.name == regionname:
 | |
|                     self._region_cache[regionname] = region
 | |
|                     return region
 | |
|             raise RuntimeError('No such region %s' % regionname)
 | |
| 
 | |
|     def get_entrance(self, entrance):
 | |
|         if isinstance(entrance, Entrance):
 | |
|             return entrance
 | |
|         try:
 | |
|             return self._entrance_cache[entrance]
 | |
|         except KeyError:
 | |
|             for region in self.regions:
 | |
|                 for exit in region.exits:
 | |
|                     if exit.name == entrance:
 | |
|                         self._entrance_cache[entrance] = exit
 | |
|                         return exit
 | |
|             raise RuntimeError('No such entrance %s' % entrance)
 | |
| 
 | |
|     def get_location(self, location):
 | |
|         if isinstance(location, Location):
 | |
|             return location
 | |
|         try:
 | |
|             return self._location_cache[location]
 | |
|         except KeyError:
 | |
|             for region in self.regions:
 | |
|                 for r_location in region.locations:
 | |
|                     if r_location.name == location:
 | |
|                         self._location_cache[location] = r_location
 | |
|                         return r_location
 | |
|         raise RuntimeError('No such location %s' % location)
 | |
| 
 | |
|     def get_all_state(self):
 | |
|         ret = CollectionState(self)
 | |
|         
 | |
|         def soft_collect(item):
 | |
|             if item.name.startswith('Progressive '):
 | |
|                 if 'Sword' in item.name:
 | |
|                     if ret.has('Golden Sword'):
 | |
|                         pass
 | |
|                     elif ret.has('Tempered Sword'):
 | |
|                         ret.prog_items.append('Golden Sword')
 | |
|                     elif ret.has('Master Sword'):
 | |
|                         ret.prog_items.append('Tempered Sword')
 | |
|                     elif ret.has('Fighter Sword'):
 | |
|                         ret.prog_items.append('Master Sword')
 | |
|                     else:
 | |
|                         ret.prog_items.append('Fighter Sword')
 | |
|                 elif 'Glove' in item.name:
 | |
|                     if ret.has('Titans Mitts'):
 | |
|                         pass
 | |
|                     elif ret.has('Power Glove'):
 | |
|                         ret.prog_items.append('Titans Mitts')
 | |
|                     else:
 | |
|                         ret.prog_items.append('Power Glove')
 | |
| 
 | |
|             elif item.advancement:
 | |
|                 ret.prog_items.append(item.name)
 | |
| 
 | |
|         for item in self.itempool:
 | |
|             soft_collect(item)
 | |
|         ret.sweep_for_events()
 | |
|         ret._clear_cache()
 | |
|         return ret
 | |
| 
 | |
|     def find_items(self, item):
 | |
|         return [location for location in self.get_locations() if location.item is not None and location.item.name == item]
 | |
| 
 | |
|     def push_item(self, location, item, collect=True):
 | |
|         if not isinstance(location, Location):
 | |
|             location = self.get_location(location)
 | |
| 
 | |
|         if location.item_rule(item):
 | |
|             location.item = item
 | |
|             item.location = location
 | |
|             if collect:
 | |
|                 self.state.collect(item, location.event)
 | |
| 
 | |
|             logging.getLogger('').debug('Placed %s at %s' % (item, location))
 | |
|         else:
 | |
|             raise RuntimeError('Cannot assign item %s to location %s.' % (item, location))
 | |
| 
 | |
|     def get_locations(self):
 | |
|         if self._cached_locations is None:
 | |
|             self._cached_locations = []
 | |
|             for region in self.regions:
 | |
|                 self._cached_locations.extend(region.locations)
 | |
|         return self._cached_locations
 | |
| 
 | |
|     def get_unfilled_locations(self):
 | |
|         return [location for location in self.get_locations() if location.item is None]
 | |
| 
 | |
|     def get_filled_locations(self):
 | |
|         return [location for location in self.get_locations() if location.item is not None]
 | |
| 
 | |
|     def get_reachable_locations(self, state=None):
 | |
|         if state is None:
 | |
|             state = self.state
 | |
|         return [location for location in self.get_locations() if state.can_reach(location)]
 | |
| 
 | |
|     def get_placeable_locations(self, state=None):
 | |
|         if state is None:
 | |
|             state = self.state
 | |
|         return [location for location in self.get_locations() if location.item is None and state.can_reach(location)]
 | |
| 
 | |
|     def unlocks_new_location(self, item):
 | |
|         temp_state = self.state.copy()
 | |
|         temp_state._clear_cache()
 | |
|         temp_state.collect(item, True)
 | |
| 
 | |
|         for location in self.get_unfilled_locations():
 | |
|             if temp_state.can_reach(location) and not self.state.can_reach(location):
 | |
|                 return True
 | |
| 
 | |
|         return False
 | |
| 
 | |
|     def can_beat_game(self, starting_state=None):
 | |
|         prog_locations = [location for location in self.get_locations() if location.item is not None and (location.item.advancement or location.event)]
 | |
| 
 | |
|         if starting_state:
 | |
|             state = starting_state.copy()
 | |
|         else:
 | |
|             state = CollectionState(self)
 | |
|         treasure_pieces_collected = 0
 | |
|         while prog_locations:
 | |
|             sphere = []
 | |
|             # build up spheres of collection radius. Everything in each sphere is independent from each other in dependencies and only depends on lower spheres
 | |
|             for location in prog_locations:
 | |
|                 if state.can_reach(location):
 | |
|                     if location.item.name == 'Triforce':
 | |
|                         return True
 | |
|                     elif location.item.name in ['Triforce Piece', 'Power Star']:
 | |
|                         treasure_pieces_collected += 1
 | |
|                     if self.goal in ['starhunt', 'triforcehunt'] and treasure_pieces_collected >= self.treasure_hunt_count:
 | |
|                         return True
 | |
|                     sphere.append(location)
 | |
| 
 | |
|             if not sphere:
 | |
|                 # ran out of places and did not find triforce yet, quit
 | |
|                 return False
 | |
| 
 | |
|             for location in sphere:
 | |
|                 prog_locations.remove(location)
 | |
|                 state.collect(location.item, True)
 | |
| 
 | |
|         return False
 | |
| 
 | |
|     @property
 | |
|     def option_identifier(self):
 | |
|         logic = 0 if self.logic == 'noglitches' else 1
 | |
|         mode = ['standard', 'open', 'swordless'].index(self.mode)
 | |
|         dungeonitems = 0 if self.place_dungeon_items else 1
 | |
|         goal = ['ganon', 'pedestal', 'dungeons', 'starhunt', 'triforcehunt'].index(self.goal)
 | |
|         shuffle = ['vanilla', 'simple', 'restricted', 'full', 'madness', 'insanity', 'dungeonsfull', 'dungeonssimple'].index(self.shuffle)
 | |
|         difficulty = ['normal', 'timed', 'timed-ohko', 'timed-countdown'].index(self.difficulty)
 | |
|         algorithm = ['freshness', 'flood', 'vt21', 'vt22', 'restrictive'].index(self.algorithm)
 | |
|         beatableonly = 1 if self.check_beatable_only else 0
 | |
|         shuffleganon = 1 if self.shuffle_ganon else 0
 | |
|         return logic | (beatableonly << 1) | (dungeonitems << 2) | (shuffleganon << 3) | (goal << 4) | (shuffle << 7) | (difficulty << 11) | (algorithm << 13) | (mode << 16)
 | |
| 
 | |
| 
 | |
| class CollectionState(object):
 | |
| 
 | |
|     def __init__(self, parent):
 | |
|         self.prog_items = []
 | |
|         self.world = parent
 | |
|         self.region_cache = {}
 | |
|         self.location_cache = {}
 | |
|         self.entrance_cache = {}
 | |
|         self.recursion_count = 0
 | |
|         self.events = []
 | |
| 
 | |
|     def _clear_cache(self):
 | |
|         # we only need to invalidate results which were False, places we could reach before we can still reach after adding more items
 | |
|         self.region_cache = {k: v for k, v in self.region_cache.items() if v}
 | |
|         self.location_cache = {k: v for k, v in self.location_cache.items() if v}
 | |
|         self.entrance_cache = {k: v for k, v in self.entrance_cache.items() if v}
 | |
| 
 | |
|     def copy(self):
 | |
|         ret = CollectionState(self.world)
 | |
|         ret.prog_items = copy.copy(self.prog_items)
 | |
|         ret.region_cache = copy.copy(self.region_cache)
 | |
|         ret.location_cache = copy.copy(self.location_cache)
 | |
|         ret.entrance_cache = copy.copy(self.entrance_cache)
 | |
|         ret.events = copy.copy(self.events)
 | |
|         return ret
 | |
| 
 | |
|     def can_reach(self, spot, resolution_hint=None):
 | |
|         try:
 | |
|             spot_type = spot.spot_type
 | |
|             if spot_type == 'Location':
 | |
|                 correct_cache = self.location_cache
 | |
|             elif spot_type == 'Region':
 | |
|                 correct_cache = self.region_cache
 | |
|             elif spot_type == 'Entrance':
 | |
|                 correct_cache = self.entrance_cache
 | |
|             else:
 | |
|                 raise AttributeError
 | |
|         except AttributeError:
 | |
|             # try to resolve a name
 | |
|             if resolution_hint == 'Location':
 | |
|                 spot = self.world.get_location(spot)
 | |
|                 correct_cache = self.location_cache
 | |
|             elif resolution_hint == 'Entrance':
 | |
|                 spot = self.world.get_entrance(spot)
 | |
|                 correct_cache = self.entrance_cache
 | |
|             else:
 | |
|                 # default to Region
 | |
|                 spot = self.world.get_region(spot)
 | |
|                 correct_cache = self.region_cache
 | |
| 
 | |
|         if spot.recursion_count > 0:
 | |
|             return False
 | |
| 
 | |
|         if spot not in correct_cache:
 | |
|             # for the purpose of evaluating results, recursion is resolved by always denying recursive access (as that ia what we are trying to figure out right now in the first place
 | |
|             spot.recursion_count += 1
 | |
|             self.recursion_count += 1
 | |
|             can_reach = spot.can_reach(self)
 | |
|             spot.recursion_count -= 1
 | |
|             self.recursion_count -= 1
 | |
| 
 | |
|             # we only store qualified false results (i.e. ones not inside a hypothetical)
 | |
|             if not can_reach:
 | |
|                 if self.recursion_count == 0:
 | |
|                     correct_cache[spot] = can_reach
 | |
|             else:
 | |
|                 correct_cache[spot] = can_reach
 | |
|             return can_reach
 | |
|         return correct_cache[spot]
 | |
| 
 | |
|     def sweep_for_events(self, key_only=False):
 | |
|         # this may need improvement
 | |
|         new_locations = True
 | |
|         checked_locations = 0
 | |
|         while new_locations:
 | |
|             reachable_events = [location for location in self.world.get_filled_locations() if location.event and (not key_only or location.item.key) and self.can_reach(location)]
 | |
|             for event in reachable_events:
 | |
|                 if event.name not in self.events:
 | |
|                     self.events.append(event.name)
 | |
|                     self.collect(event.item, True)
 | |
|             new_locations = len(reachable_events) > checked_locations
 | |
|             checked_locations = len(reachable_events)
 | |
| 
 | |
|     def has(self, item, count=1):
 | |
|         if count == 1:
 | |
|             return item in self.prog_items
 | |
|         else:
 | |
|             return len([pritem for pritem in self.prog_items if pritem == item]) >= count
 | |
| 
 | |
|     def can_lift_rocks(self):
 | |
|         return self.has('Power Glove') or self.has('Titans Mitts')
 | |
| 
 | |
|     def can_lift_heavy_rocks(self):
 | |
|         return self.has('Titans Mitts')
 | |
| 
 | |
|     def has_sword(self):
 | |
|         return self.has('Fighter Sword') or self.has('Master Sword') or self.has('Tempered Sword') or self.has('Golden Sword')
 | |
| 
 | |
|     def has_beam_sword(self):
 | |
|         return self.has('Master Sword') or self.has('Tempered Sword') or self.has('Golden Sword')
 | |
| 
 | |
|     def has_blunt_weapon(self):
 | |
|         return self.has_sword() or self.has('Hammer')
 | |
| 
 | |
|     def has_Mirror(self):
 | |
|         return self.has('Magic Mirror')
 | |
| 
 | |
|     def has_Boots(self):
 | |
|         return self.has('Pegasus Boots')
 | |
| 
 | |
|     def has_Pearl(self):
 | |
|         return self.has('Moon Pearl')
 | |
| 
 | |
|     def has_fire_source(self):
 | |
|         return self.has('Fire Rod') or self.has('Lamp')
 | |
| 
 | |
|     def has_misery_mire_medallion(self):
 | |
|         return self.has(self.world.required_medallions[0])
 | |
| 
 | |
|     def has_turtle_rock_medallion(self):
 | |
|         return self.has(self.world.required_medallions[1])
 | |
| 
 | |
|     def collect(self, item, event=False):
 | |
|         changed = False
 | |
|         if item.name.startswith('Progressive '):
 | |
|             if 'Sword' in item.name:
 | |
|                 if self.has('Golden Sword'):
 | |
|                     pass
 | |
|                 elif self.has('Tempered Sword'):
 | |
|                     self.prog_items.append('Golden Sword')
 | |
|                     changed = True
 | |
|                 elif self.has('Master Sword'):
 | |
|                     self.prog_items.append('Tempered Sword')
 | |
|                     changed = True
 | |
|                 elif self.has('Fighter Sword'):
 | |
|                     self.prog_items.append('Master Sword')
 | |
|                     changed = True
 | |
|                 else:
 | |
|                     self.prog_items.append('Fighter Sword')
 | |
|                     changed = True
 | |
|             elif 'Glove' in item.name:
 | |
|                 if self.has('Titans Mitts'):
 | |
|                     pass
 | |
|                 elif self.has('Power Glove'):
 | |
|                     self.prog_items.append('Titans Mitts')
 | |
|                     changed = True
 | |
|                 else:
 | |
|                     self.prog_items.append('Power Glove')
 | |
|                     changed = True
 | |
| 
 | |
|         elif event or item.advancement:
 | |
|             self.prog_items.append(item.name)
 | |
|             changed = True
 | |
| 
 | |
|         if changed:
 | |
|             self._clear_cache()
 | |
|             if not event:
 | |
|                 self.sweep_for_events()
 | |
|                 self._clear_cache()
 | |
| 
 | |
|     def remove(self, item):
 | |
|         if item.advancement:
 | |
|             to_remove = item.name
 | |
|             if to_remove.startswith('Progressive '):
 | |
|                 if 'Sword' in to_remove:
 | |
|                     if self.has('Golden Sword'):
 | |
|                         to_remove = 'Golden Sword'
 | |
|                     elif self.has('Tempered Sword'):
 | |
|                         to_remove = 'Tempered Sword'
 | |
|                     elif self.has('Master Sword'):
 | |
|                         to_remove = 'Master Sword'
 | |
|                     elif self.has('Fighter Sword'):
 | |
|                         to_remove = 'Fighter Sword'
 | |
|                     else:
 | |
|                         to_remove = None
 | |
|                 elif 'Glove' in item.name:
 | |
|                     if self.has('Titans Mitts'):
 | |
|                         to_remove = 'Titans Mitts'
 | |
|                     elif self.has('Power Glove'):
 | |
|                         to_remove = 'Power Glove'
 | |
|                     else:
 | |
|                         to_remove = None
 | |
| 
 | |
|             if to_remove is not None:
 | |
|                 try:
 | |
|                     self.prog_items.remove(to_remove)
 | |
|                 except ValueError:
 | |
|                     return
 | |
| 
 | |
|                 # invalidate caches, nothing can be trusted anymore now
 | |
|                 self.region_cache = {}
 | |
|                 self.location_cache = {}
 | |
|                 self.entrance_cache = {}
 | |
|                 self.recursion_count = 0
 | |
| 
 | |
|     def __getattr__(self, item):
 | |
|         if item.startswith('can_reach_'):
 | |
|             return self.can_reach(item[10])
 | |
|         elif item.startswith('has_'):
 | |
|             return self.has(item[4])
 | |
| 
 | |
|         raise RuntimeError('Cannot parse %s.' % item)
 | |
| 
 | |
| 
 | |
| class Region(object):
 | |
| 
 | |
|     def __init__(self, name):
 | |
|         self.name = name
 | |
|         self.entrances = []
 | |
|         self.exits = []
 | |
|         self.locations = []
 | |
|         self.spot_type = 'Region'
 | |
|         self.hint_text = 'Hyrule'
 | |
|         self.recursion_count = 0
 | |
| 
 | |
|     def can_reach(self, state):
 | |
|         for entrance in self.entrances:
 | |
|             if state.can_reach(entrance):
 | |
|                 return True
 | |
|         return False
 | |
| 
 | |
|     def __str__(self):
 | |
|         return str(self.__unicode__())
 | |
| 
 | |
|     def __unicode__(self):
 | |
|         return '%s' % self.name
 | |
| 
 | |
| 
 | |
| class Entrance(object):
 | |
| 
 | |
|     def __init__(self, name='', parent=None):
 | |
|         self.name = name
 | |
|         self.parent_region = parent
 | |
|         self.connected_region = None
 | |
|         self.target = None
 | |
|         self.addresses = None
 | |
|         self.spot_type = 'Entrance'
 | |
|         self.recursion_count = 0
 | |
|         self.vanilla = None
 | |
| 
 | |
|     def access_rule(self, state):
 | |
|         return True
 | |
| 
 | |
|     def can_reach(self, state):
 | |
|         if self.access_rule(state) and state.can_reach(self.parent_region):
 | |
|             return True
 | |
| 
 | |
|         return False
 | |
| 
 | |
|     def connect(self, region, addresses=None, target=None, vanilla=None):
 | |
|         self.connected_region = region
 | |
|         self.target = target
 | |
|         self.addresses = addresses
 | |
|         self.vanilla = vanilla
 | |
|         region.entrances.append(self)
 | |
| 
 | |
|     def __str__(self):
 | |
|         return str(self.__unicode__())
 | |
| 
 | |
|     def __unicode__(self):
 | |
|         return '%s' % self.name
 | |
| 
 | |
| 
 | |
| class Location(object):
 | |
| 
 | |
|     def __init__(self, name='', address=None, crystal=False, hint_text=None, parent=None):
 | |
|         self.name = name
 | |
|         self.parent_region = parent
 | |
|         self.item = None
 | |
|         self.crystal = crystal
 | |
|         self.address = address
 | |
|         self.spot_type = 'Location'
 | |
|         self.hint_text = hint_text if hint_text is not None else 'Hyrule'
 | |
|         self.recursion_count = 0
 | |
|         self.staleness_count = 0
 | |
|         self.event = False
 | |
| 
 | |
|     def access_rule(self, state):
 | |
|         return True
 | |
| 
 | |
|     def item_rule(self, item):
 | |
|         return True
 | |
| 
 | |
|     def can_reach(self, state):
 | |
|         if self.access_rule(state) and state.can_reach(self.parent_region):
 | |
|             return True
 | |
|         return False
 | |
| 
 | |
|     def __str__(self):
 | |
|         return str(self.__unicode__())
 | |
| 
 | |
|     def __unicode__(self):
 | |
|         return '%s' % self.name
 | |
| 
 | |
| 
 | |
| class Item(object):    
 | |
| 
 | |
|     def __init__(self, name='', advancement=False, priority=False, key=False, crystal=False, code=None, altar_hint=None, altar_credit=None, sickkid_credit=None, zora_credit=None, witch_credit=None, fluteboy_credit=None):
 | |
|         self.name = name
 | |
|         self.advancement = advancement
 | |
|         self.priority = priority
 | |
|         self.key = key
 | |
|         self.crystal = crystal
 | |
|         self.altar_hint_text = altar_hint
 | |
|         self.altar_credit_text = altar_credit
 | |
|         self.sickkid_credit_text = sickkid_credit
 | |
|         self.zora_credit_text = zora_credit
 | |
|         self.magicshop_credit_text = witch_credit
 | |
|         self.fluteboy_credit_text = fluteboy_credit
 | |
|         self.code = code
 | |
|         self.location = None
 | |
| 
 | |
|     def __str__(self):
 | |
|         return str(self.__unicode__())
 | |
| 
 | |
|     def __unicode__(self):
 | |
|         return '%s' % self.name
 | |
| 
 | |
| 
 | |
| # have 6 address that need to be filled
 | |
| class Crystal(Item):
 | |
|     pass
 | |
| 
 | |
| 
 | |
| class Spoiler(object):
 | |
| 
 | |
|     def __init__(self, world):
 | |
|         self.world = world
 | |
|         self.entrances = []
 | |
|         self.medallions = {}
 | |
|         self.playthrough = {}
 | |
|         self.locations = {}
 | |
|         self.metadata = {}
 | |
| 
 | |
|     def set_entrance(self, entrance, exit, direction):
 | |
|         self.entrances.append(OrderedDict([('entrance', entrance), ('exit', exit), ('direction', direction)]))
 | |
| 
 | |
|     def parse_data(self):
 | |
|         self.medallions = OrderedDict([('Misery Mire', self.world.required_medallions[0]), ('Turtle Rock', self.world.required_medallions[1])])
 | |
|         self.locations = {'other locations': OrderedDict([(str(location), str(location.item)) if location.item is not None else 'Nothing' for location in self.world.get_locations()])}
 | |
|         from Main import __version__ as ERVersion
 | |
|         self.metadata = {'version': ERVersion,
 | |
|                          'seed': self.world.seed,
 | |
|                          'logic': self.world.logic,
 | |
|                          'mode': self.world.mode,
 | |
|                          'goal': self.world.goal,
 | |
|                          'shuffle': self.world.shuffle,
 | |
|                          'algorithm': self.world.algorithm}
 | |
| 
 | |
|     def to_json(self):
 | |
|         self.parse_data()
 | |
|         out = OrderedDict()
 | |
|         out['entrances'] = self.entrances
 | |
|         out.update(self.locations)
 | |
|         out['medallions'] = self.medallions
 | |
|         out['playthrough'] = self.playthrough
 | |
|         out['meta'] = self.metadata
 | |
|         return json.dumps(out)
 | |
| 
 | |
|     def to_file(self, filename):
 | |
|         self.parse_data()
 | |
|         with open(filename, 'w') as outfile:
 | |
|             outfile.write('ALttP Entrance Randomizer Version %s  -  Seed: %s\n\n' % (self.metadata['version'], self.metadata['seed']))
 | |
|             outfile.write('Logic: %s  Mode: %s  Goal: %s  Entrance Shuffle: %s  Filling Algorithm: %s' % (self.metadata['logic'], self.metadata['mode'], self.metadata['goal'], self.metadata['shuffle'], self.metadata['algorithm']))
 | |
|             if self.entrances:
 | |
|                 outfile.write('\n\nEntrances:\n\n')
 | |
|                 outfile.write('\n'.join(['%s %s %s' % (entry['entrance'], '<=>' if entry['direction'] == 'both' else '<=' if entry['direction'] == 'exit' else '=>', entry['exit']) for entry in self.entrances]))
 | |
|             outfile.write('\n\nMedallions')
 | |
|             outfile.write('\n\nMisery Mire Medallion: %s' % self.medallions['Misery Mire'])
 | |
|             outfile.write('\nTurtle Rock Medallion: %s' % self.medallions['Turtle Rock'])
 | |
|             outfile.write('\n\nLocations:\n\n')
 | |
|             outfile.write('\n'.join(['%s: %s' % (location, item) for (location, item) in self.locations['other locations'].items()]))
 | |
|             outfile.write('\n\nPlaythrough:\n\n')
 | |
|             outfile.write('\n'.join(['%s: {\n%s\n}' % (sphere_nr, '\n'.join(['  %s: %s' % (location, item) for (location, item) in sphere.items()])) for (sphere_nr, sphere) in self.playthrough.items()]))
 | 
