2017-05-25 15:58:35 +02:00
from BaseClasses import World , CollectionState , Item
2017-05-25 12:09:50 +02:00
from Regions import create_regions
from EntranceShuffle import link_entrances
2017-07-14 14:37:34 +02:00
from Rom import patch_rom , LocalRom , JsonRom
2017-05-15 20:28:04 +02:00
from Rules import set_rules
from Dungeons import fill_dungeons
2017-05-25 15:58:35 +02:00
from Items import ItemFactory
2017-07-18 12:44:13 +02:00
from collections import OrderedDict
2017-05-15 20:28:04 +02:00
import random
import time
2017-05-16 21:23:47 +02:00
import logging
2017-07-16 23:20:54 +02:00
import json
2017-05-15 20:28:04 +02:00
2017-07-20 19:12:55 +02:00
__version__ = ' 0.4.5-dev '
2017-05-20 14:07:40 +02:00
2017-07-20 19:12:55 +02:00
logic_hash = [ 100 , 18 , 94 , 177 , 161 , 252 , 4 , 45 , 29 , 231 , 99 , 158 , 70 , 55 , 74 , 39 , 12 , 134 , 142 , 189 , 61 , 105 , 10 , 254 , 137 , 44 , 72 , 154 , 145 , 167 , 98 , 225 ,
2017-07-20 11:22:35 +02:00
215 , 217 , 126 , 187 , 13 , 255 , 138 , 51 , 64 , 130 , 139 , 233 , 168 , 69 , 175 , 25 , 58 , 160 , 1 , 27 , 206 , 169 , 223 , 210 , 188 , 111 , 186 , 240 , 133 , 26 , 41 , 241 ,
2017-06-23 22:15:29 +02:00
204 , 89 , 78 , 63 , 96 , 218 , 198 , 224 , 219 , 35 , 82 , 181 , 121 , 243 , 0 , 155 , 91 , 120 , 221 , 178 , 162 , 80 , 66 , 97 , 118 , 103 , 86 , 191 , 135 , 122 , 104 , 40 ,
183 , 9 , 230 , 110 , 14 , 87 , 143 , 249 , 90 , 75 , 232 , 157 , 238 , 196 , 23 , 248 , 2 , 101 , 159 , 108 , 201 , 73 , 34 , 15 , 179 , 92 , 226 , 60 , 222 , 32 , 109 , 119 ,
2017-07-18 13:42:33 +02:00
49 , 56 , 16 , 6 , 22 , 209 , 190 , 21 , 136 , 205 , 113 , 192 , 146 , 30 , 212 , 43 , 200 , 193 , 185 , 242 , 71 , 163 , 102 , 239 , 24 , 220 , 166 , 228 , 208 , 47 , 3 , 112 ,
2017-06-23 22:15:29 +02:00
203 , 50 , 216 , 214 , 107 , 106 , 57 , 67 , 88 , 42 , 176 , 129 , 144 , 54 , 237 , 165 , 116 , 141 , 125 , 128 , 172 , 171 , 152 , 83 , 38 , 93 , 148 , 151 , 207 , 236 , 131 , 85 ,
170 , 124 , 28 , 251 , 194 , 250 , 8 , 164 , 65 , 20 , 150 , 182 , 77 , 17 , 202 , 253 , 173 , 229 , 46 , 140 , 76 , 95 , 117 , 174 , 79 , 84 , 36 , 244 , 199 , 37 , 211 , 7 ,
247 , 213 , 31 , 62 , 59 , 153 , 197 , 19 , 48 , 114 , 53 , 115 , 149 , 81 , 5 , 184 , 147 , 68 , 227 , 234 , 52 , 156 , 132 , 127 , 235 , 245 , 11 , 33 , 123 , 180 , 246 , 195 ]
2017-05-25 21:41:37 +02:00
2017-05-20 14:07:40 +02:00
2017-05-25 15:58:35 +02:00
def main ( args , seed = None ) :
2017-05-20 14:07:40 +02:00
start = time . clock ( )
2017-05-15 20:28:04 +02:00
# initialize the world
2017-07-17 22:28:29 +02:00
world = World ( args . shuffle , args . logic , args . mode , args . difficulty , args . goal , args . algorithm , not args . nodungeonitems , args . beatableonly , args . shuffleganon )
2017-05-20 14:07:40 +02:00
logger = logging . getLogger ( ' ' )
if seed is None :
random . seed ( None )
world . seed = random . randint ( 0 , 999999999 )
else :
2017-05-30 07:33:23 +02:00
world . seed = int ( seed )
2017-05-20 14:07:40 +02:00
random . seed ( world . seed )
2017-07-18 12:44:13 +02:00
logger . info ( ' ALttP Entrance Randomizer Version %s - Seed: %s \n \n ' % ( __version__ , world . seed ) )
2017-05-20 14:07:40 +02:00
2017-05-15 20:28:04 +02:00
create_regions ( world )
2017-05-20 14:07:40 +02:00
logger . info ( ' Shuffling the World about. ' )
2017-07-18 12:44:13 +02:00
link_entrances ( world )
2017-05-20 14:07:40 +02:00
2017-06-23 21:32:31 +02:00
logger . info ( ' Generating Item Pool. ' )
2017-07-18 12:44:13 +02:00
generate_itempool ( world )
2017-06-23 21:32:31 +02:00
2017-05-20 14:07:40 +02:00
logger . info ( ' Calculating Access Rules. ' )
2017-07-18 12:44:13 +02:00
set_rules ( world )
2017-05-20 14:07:40 +02:00
2017-06-23 21:32:31 +02:00
logger . info ( ' Placing Dungeon Items. ' )
2017-05-20 14:07:40 +02:00
2017-07-18 12:44:13 +02:00
fill_dungeons ( world )
2017-05-20 14:07:40 +02:00
logger . info ( ' Fill the world. ' )
2017-05-25 15:58:35 +02:00
if args . algorithm == ' flood ' :
2017-05-20 14:07:40 +02:00
flood_items ( world ) # different algo, biased towards early game progress items
2017-06-03 21:28:02 +02:00
elif args . algorithm == ' vt21 ' :
distribute_items_cutoff ( world , 1 )
elif args . algorithm == ' vt22 ' :
2017-06-04 16:15:59 +02:00
distribute_items_cutoff ( world , 0.66 )
2017-06-03 21:28:02 +02:00
elif args . algorithm == ' freshness ' :
distribute_items_staleness ( world )
2017-06-19 21:31:08 +02:00
elif args . algorithm == ' restrictive ' :
2017-07-20 11:22:35 +02:00
distribute_items_restrictive ( world , 0 )
2017-06-03 21:28:02 +02:00
2017-05-20 14:07:40 +02:00
logger . info ( ' Calculating playthrough. ' )
2017-07-18 12:44:13 +02:00
create_playthrough ( world )
2017-05-20 14:07:40 +02:00
logger . info ( ' Patching ROM. ' )
2017-05-26 18:39:32 +02:00
if args . sprite is not None :
sprite = bytearray ( open ( args . sprite , ' rb ' ) . read ( ) )
else :
sprite = None
2017-06-04 15:02:27 +02:00
outfilebase = ' ER_ %s _ %s _ %s _ %s _ %s _ %s ' % ( world . mode , world . goal , world . shuffle , world . difficulty , world . algorithm , world . seed )
2017-05-20 14:07:40 +02:00
2017-06-04 13:09:47 +02:00
if not args . suppress_rom :
2017-07-14 14:37:34 +02:00
if args . jsonout :
rom = JsonRom ( )
else :
rom = LocalRom ( args . rom )
patch_rom ( world , rom , bytearray ( logic_hash ) , args . quickswap , args . heartbeep , sprite )
2017-07-16 23:20:54 +02:00
if args . jsonout :
2017-07-18 12:44:13 +02:00
print ( json . dumps ( { ' patch ' : rom . patches , ' spoiler ' : world . spoiler . to_json ( ) } ) )
2017-07-16 23:20:54 +02:00
else :
rom . write_to_file ( args . jsonout or ' %s .sfc ' % outfilebase )
2017-06-04 13:09:47 +02:00
2017-07-16 23:20:54 +02:00
if args . create_spoiler and not args . jsonout :
2017-07-18 12:44:13 +02:00
world . spoiler . to_file ( ' %s _Spoiler.txt ' % outfilebase )
2017-05-20 14:07:40 +02:00
logger . info ( ' Done. Enjoy. ' )
logger . debug ( ' Total Time: %s ' % ( time . clock ( ) - start ) )
2017-05-15 20:28:04 +02:00
return world
2017-06-03 21:28:02 +02:00
def distribute_items_cutoff ( world , cutoffrate = 0.33 ) :
# get list of locations to fill in
fill_locations = world . get_unfilled_locations ( )
random . shuffle ( fill_locations )
# get items to distribute
random . shuffle ( world . itempool )
itempool = world . itempool
total_advancement_items = len ( [ item for item in itempool if item . advancement ] )
placed_advancement_items = 0
progress_done = False
2017-06-04 13:09:11 +02:00
advancement_placed = False
2017-06-03 21:28:02 +02:00
2017-06-24 11:11:56 +02:00
# sweep once to pick up preplaced items
world . state . sweep_for_events ( )
2017-06-03 21:28:02 +02:00
while itempool and fill_locations :
candidate_item_to_place = None
item_to_place = None
for item in itempool :
2017-06-04 14:43:13 +02:00
if advancement_placed or ( progress_done and ( item . advancement or item . priority ) ) :
2017-06-03 21:28:02 +02:00
item_to_place = item
break
if item . advancement :
candidate_item_to_place = item
if world . unlocks_new_location ( item ) :
item_to_place = item
placed_advancement_items + = 1
break
if item_to_place is None :
# check if we can reach all locations and that is why we find no new locations to place
2017-06-04 13:09:11 +02:00
if not progress_done and len ( world . get_reachable_locations ( ) ) == len ( world . get_locations ( ) ) :
2017-06-03 21:28:02 +02:00
progress_done = True
continue
2017-06-04 13:09:11 +02:00
# check if we have now placed all advancement items
if progress_done :
advancement_placed = True
continue
2017-06-03 21:28:02 +02:00
# we might be in a situation where all new locations require multiple items to reach. If that is the case, just place any advancement item we've found and continue trying
if candidate_item_to_place is not None :
item_to_place = candidate_item_to_place
placed_advancement_items + = 1
else :
# we placed all available progress items. Maybe the game can be beaten anyway?
if world . can_beat_game ( ) :
logging . getLogger ( ' ' ) . warning ( ' Not all locations reachable. Game beatable anyway. ' )
progress_done = True
continue
raise RuntimeError ( ' No more progress items left to place. ' )
spot_to_fill = None
for location in ( fill_locations if placed_advancement_items / total_advancement_items < cutoffrate else reversed ( fill_locations ) ) :
if world . state . can_reach ( location ) and location . item_rule ( item_to_place ) :
spot_to_fill = location
break
if spot_to_fill is None :
# we filled all reachable spots. Maybe the game can be beaten anyway?
if world . can_beat_game ( ) :
logging . getLogger ( ' ' ) . warning ( ' Not all items placed. Game beatable anyway. ' )
break
raise RuntimeError ( ' No more spots to place %s ' % item_to_place )
world . push_item ( spot_to_fill , item_to_place , True )
itempool . remove ( item_to_place )
fill_locations . remove ( spot_to_fill )
logging . getLogger ( ' ' ) . debug ( ' Unplaced items: %s - Unfilled Locations: %s ' % ( [ item . name for item in itempool ] , [ location . name for location in fill_locations ] ) )
def distribute_items_staleness ( world ) :
2017-05-15 20:28:04 +02:00
# get list of locations to fill in
fill_locations = world . get_unfilled_locations ( )
random . shuffle ( fill_locations )
# get items to distribute
random . shuffle ( world . itempool )
itempool = world . itempool
progress_done = False
2017-06-04 13:09:11 +02:00
advancement_placed = False
2017-05-15 20:28:04 +02:00
2017-06-24 11:11:56 +02:00
# sweep once to pick up preplaced items
world . state . sweep_for_events ( )
2017-05-15 20:28:04 +02:00
while itempool and fill_locations :
candidate_item_to_place = None
item_to_place = None
for item in itempool :
2017-06-04 14:43:13 +02:00
if advancement_placed or ( progress_done and ( item . advancement or item . priority ) ) :
2017-05-15 20:28:04 +02:00
item_to_place = item
break
if item . advancement :
candidate_item_to_place = item
if world . unlocks_new_location ( item ) :
item_to_place = item
break
if item_to_place is None :
# check if we can reach all locations and that is why we find no new locations to place
2017-06-04 13:09:11 +02:00
if not progress_done and len ( world . get_reachable_locations ( ) ) == len ( world . get_locations ( ) ) :
2017-05-15 20:28:04 +02:00
progress_done = True
continue
2017-06-04 13:09:11 +02:00
# check if we have now placed all advancement items
if progress_done :
advancement_placed = True
continue
2017-05-15 20:28:04 +02:00
# we might be in a situation where all new locations require multiple items to reach. If that is the case, just place any advancement item we've found and continue trying
if candidate_item_to_place is not None :
item_to_place = candidate_item_to_place
else :
2017-05-16 21:23:47 +02:00
# we placed all available progress items. Maybe the game can be beaten anyway?
if world . can_beat_game ( ) :
logging . getLogger ( ' ' ) . warning ( ' Not all locations reachable. Game beatable anyway. ' )
2017-05-22 21:34:38 +02:00
progress_done = True
continue
2017-05-15 20:28:04 +02:00
raise RuntimeError ( ' No more progress items left to place. ' )
spot_to_fill = None
for location in fill_locations :
2017-06-03 21:28:02 +02:00
# increase likelyhood of skipping a location if it has been found stale
if not progress_done and random . randint ( 0 , location . staleness_count ) > 2 :
continue
2017-05-15 20:28:04 +02:00
if world . state . can_reach ( location ) and location . item_rule ( item_to_place ) :
spot_to_fill = location
break
2017-06-03 21:28:02 +02:00
else :
location . staleness_count + = 1
# might have skipped too many locations due to potential staleness. Do not check for staleness now to find a candidate
if spot_to_fill is None :
for location in fill_locations :
if world . state . can_reach ( location ) and location . item_rule ( item_to_place ) :
spot_to_fill = location
break
2017-05-15 20:28:04 +02:00
if spot_to_fill is None :
2017-05-16 21:23:47 +02:00
# we filled all reachable spots. Maybe the game can be beaten anyway?
if world . can_beat_game ( ) :
logging . getLogger ( ' ' ) . warning ( ' Not all items placed. Game beatable anyway. ' )
break
2017-05-15 20:28:04 +02:00
raise RuntimeError ( ' No more spots to place %s ' % item_to_place )
world . push_item ( spot_to_fill , item_to_place , True )
itempool . remove ( item_to_place )
fill_locations . remove ( spot_to_fill )
2017-05-26 09:55:24 +02:00
logging . getLogger ( ' ' ) . debug ( ' Unplaced items: %s - Unfilled Locations: %s ' % ( [ item . name for item in itempool ] , [ location . name for location in fill_locations ] ) )
2017-05-16 21:23:47 +02:00
2017-06-23 22:15:29 +02:00
def distribute_items_restrictive ( world , gftower_trash_count = 0 ) :
2017-06-19 21:31:08 +02:00
# get list of locations to fill in
fill_locations = world . get_unfilled_locations ( )
# get items to distribute
random . shuffle ( world . itempool )
progitempool = [ item for item in world . itempool if item . advancement ]
prioitempool = [ item for item in world . itempool if not item . advancement and item . priority ]
restitempool = [ item for item in world . itempool if not item . advancement and not item . priority ]
2017-06-23 19:07:34 +02:00
# fill in gtower locations with trash first
gtower_locations = [ location for location in fill_locations if ' Ganons Tower ' in location . name ]
random . shuffle ( gtower_locations )
trashcnt = 0
while gtower_locations and restitempool and trashcnt < gftower_trash_count :
spot_to_fill = gtower_locations . pop ( )
item_to_place = restitempool . pop ( )
world . push_item ( spot_to_fill , item_to_place , False )
fill_locations . remove ( spot_to_fill )
trashcnt + = 1
random . shuffle ( fill_locations )
def sweep_from_pool ( ) :
2017-06-19 21:31:08 +02:00
new_state = world . state . copy ( )
for item in progitempool :
new_state . collect ( item , True )
new_state . sweep_for_events ( )
return new_state
while progitempool and fill_locations :
item_to_place = progitempool . pop ( )
2017-06-23 19:07:34 +02:00
maximum_exploration_state = sweep_from_pool ( )
2017-06-19 21:31:08 +02:00
spot_to_fill = None
for location in fill_locations :
2017-06-23 22:15:29 +02:00
if location . item_rule ( item_to_place ) :
if world . check_beatable_only :
starting_state = world . state . copy ( )
for item in progitempool :
starting_state . collect ( item , True )
if maximum_exploration_state . can_reach ( location ) :
if world . check_beatable_only :
starting_state . collect ( item_to_place , True )
else :
spot_to_fill = location
break
if world . check_beatable_only and world . can_beat_game ( starting_state ) :
spot_to_fill = location
break
2017-06-19 21:31:08 +02:00
if spot_to_fill is None :
# we filled all reachable spots. Maybe the game can be beaten anyway?
if world . can_beat_game ( ) :
2017-06-23 22:15:29 +02:00
if not world . check_beatable_only :
logging . getLogger ( ' ' ) . warning ( ' Not all items placed. Game beatable anyway. ' )
2017-06-19 21:31:08 +02:00
break
raise RuntimeError ( ' No more spots to place %s ' % item_to_place )
world . push_item ( spot_to_fill , item_to_place , False )
fill_locations . remove ( spot_to_fill )
spot_to_fill . event = True
2017-06-23 19:07:34 +02:00
random . shuffle ( fill_locations )
2017-06-19 21:31:08 +02:00
while prioitempool and fill_locations :
spot_to_fill = fill_locations . pop ( )
item_to_place = prioitempool . pop ( )
world . push_item ( spot_to_fill , item_to_place , False )
while restitempool and fill_locations :
spot_to_fill = fill_locations . pop ( )
item_to_place = restitempool . pop ( )
world . push_item ( spot_to_fill , item_to_place , False )
logging . getLogger ( ' ' ) . debug ( ' Unplaced items: %s - Unfilled Locations: %s ' % ( [ item . name for item in progitempool + prioitempool + restitempool ] , [ location . name for location in fill_locations ] ) )
2017-05-16 21:23:47 +02:00
def flood_items ( world ) :
# get items to distribute
random . shuffle ( world . itempool )
itempool = world . itempool
progress_done = False
2017-06-24 11:11:56 +02:00
# sweep once to pick up preplaced items
world . state . sweep_for_events ( )
2017-05-16 21:23:47 +02:00
# fill world from top of itempool while we can
while not progress_done :
location_list = world . get_unfilled_locations ( )
random . shuffle ( location_list )
spot_to_fill = None
for location in location_list :
2017-06-04 18:32:10 +02:00
if world . state . can_reach ( location ) and location . item_rule ( itempool [ 0 ] ) :
2017-05-16 21:23:47 +02:00
spot_to_fill = location
break
if spot_to_fill :
item = itempool . pop ( 0 )
world . push_item ( spot_to_fill , item , True )
continue
# ran out of spots, check if we need to step in and correct things
if len ( world . get_reachable_locations ( ) ) == len ( world . get_locations ( ) ) :
progress_done = True
continue
2017-05-15 20:28:04 +02:00
2017-05-16 21:23:47 +02:00
# need to place a progress item instead of an already placed item, find candidate
item_to_place = None
candidate_item_to_place = None
for item in itempool :
if item . advancement :
candidate_item_to_place = item
if world . unlocks_new_location ( item ) :
item_to_place = item
break
# we might be in a situation where all new locations require multiple items to reach. If that is the case, just place any advancement item we've found and continue trying
if item_to_place is None :
if candidate_item_to_place is not None :
item_to_place = candidate_item_to_place
else :
raise RuntimeError ( ' No more progress items left to place. ' )
2017-05-15 20:28:04 +02:00
2017-05-16 21:23:47 +02:00
# find item to replace with progress item
location_list = world . get_reachable_locations ( )
random . shuffle ( location_list )
for location in location_list :
2017-06-04 14:43:13 +02:00
if location . item is not None and not location . item . advancement and not location . item . priority and not location . item . key :
2017-05-16 21:23:47 +02:00
# safe to replace
replace_item = location . item
replace_item . location = None
itempool . append ( replace_item )
world . push_item ( location , item_to_place , True )
itempool . remove ( item_to_place )
break
def generate_itempool ( world ) :
2017-06-24 18:48:03 +02:00
if world . difficulty not in [ ' normal ' , ' timed ' , ' timed-ohko ' , ' timed-countdown ' ] or world . goal not in [ ' ganon ' , ' pedestal ' , ' dungeons ' , ' starhunt ' , ' triforcehunt ' ] or world . mode not in [ ' open ' , ' standard ' , ' swordless ' ] :
2017-05-15 20:28:04 +02:00
raise NotImplementedError ( ' Not supported yet ' )
2017-05-25 15:58:35 +02:00
world . push_item ( ' Ganon ' , ItemFactory ( ' Triforce ' ) , False )
2017-06-17 14:40:37 +02:00
world . push_item ( ' Agahnim 1 ' , ItemFactory ( ' Beat Agahnim 1 ' ) , False )
world . get_location ( ' Agahnim 1 ' ) . event = True
world . push_item ( ' Agahnim 2 ' , ItemFactory ( ' Beat Agahnim 2 ' ) , False )
world . get_location ( ' Agahnim 2 ' ) . event = True
2017-05-15 20:28:04 +02:00
# set up item pool
2017-06-04 14:44:23 +02:00
if world . difficulty in [ ' timed ' , ' timed-countdown ' ] :
world . itempool = ItemFactory ( [ ' Arrow Upgrade (+5) ' ] * 2 + [ ' Bomb Upgrade (+5) ' ] * 2 + [ ' Arrow Upgrade (+10) ' ] * 3 + [ ' Bomb Upgrade (+10) ' ] * 3 +
2017-06-24 18:48:03 +02:00
[ ' Progressive Armor ' ] * 2 + [ ' Progressive Shield ' ] * 3 + [ ' Progressive Glove ' ] * 2 +
2017-06-04 14:44:23 +02:00
[ ' Bottle ' ] * 4 +
[ ' Bombos ' , ' Book of Mudora ' , ' Blue Boomerang ' , ' Bow ' , ' Bug Catching Net ' , ' Cane of Byrna ' , ' Cane of Somaria ' ,
' Ether ' , ' Fire Rod ' , ' Flippers ' , ' Ocarina ' , ' Hammer ' , ' Hookshot ' , ' Ice Rod ' , ' Lamp ' , ' Cape ' , ' Magic Powder ' ,
' Red Boomerang ' , ' Mushroom ' , ' Pegasus Boots ' , ' Quake ' , ' Shovel ' , ' Silver Arrows ' ] +
[ ' Sanctuary Heart Container ' ] + [ ' Rupees (100) ' ] * 2 + [ ' Boss Heart Container ' ] * 12 + [ ' Piece of Heart ' ] * 16 +
2017-08-01 17:25:08 +02:00
[ ' Rupees (50) ' ] * 8 + [ ' Rupees (300) ' ] * 6 + [ ' Rupees (20) ' ] * 4 +
[ ' Arrows (10) ' ] * 3 + [ ' Bombs (3) ' ] * 10 + [ ' Red Clock ' ] * 10 + [ ' Blue Clock ' ] * 10 + [ ' Green Clock ' ] * 20 )
2017-06-04 14:44:23 +02:00
world . clock_mode = ' stopwatch ' if world . difficulty == ' timed ' else ' countdown '
elif world . difficulty == ' timed-ohko ' :
world . itempool = ItemFactory ( [ ' Arrow Upgrade (+5) ' ] * 6 + [ ' Bomb Upgrade (+5) ' ] * 6 + [ ' Arrow Upgrade (+10) ' , ' Bomb Upgrade (+10) ' ] +
2017-06-24 18:48:03 +02:00
[ ' Progressive Armor ' ] * 2 + [ ' Progressive Shield ' ] * 3 + [ ' Progressive Glove ' ] * 2 +
2017-06-04 14:44:23 +02:00
[ ' Bottle ' ] * 4 +
[ ' Bombos ' , ' Book of Mudora ' , ' Blue Boomerang ' , ' Bow ' , ' Bug Catching Net ' , ' Cane of Byrna ' , ' Cane of Somaria ' ,
' Ether ' , ' Fire Rod ' , ' Flippers ' , ' Ocarina ' , ' Hammer ' , ' Hookshot ' , ' Ice Rod ' , ' Lamp ' , ' Cape ' , ' Magic Powder ' ,
' Red Boomerang ' , ' Mushroom ' , ' Pegasus Boots ' , ' Quake ' , ' Shovel ' , ' Silver Arrows ' ] +
[ ' Single Arrow ' , ' Sanctuary Heart Container ' ] + [ ' Rupees (100) ' ] * 3 + [ ' Boss Heart Container ' ] * 10 + [ ' Piece of Heart ' ] * 24 +
2017-08-01 17:25:08 +02:00
[ ' Rupees (50) ' ] * 7 + [ ' Rupees (300) ' ] * 7 + [ ' Rupees (20) ' ] * 5 +
[ ' Arrows (10) ' ] * 5 + [ ' Bombs (3) ' ] * 10 + [ ' Green Clock ' ] * 25 )
2017-06-04 14:44:23 +02:00
world . clock_mode = ' ohko '
else :
world . itempool = ItemFactory ( [ ' Arrow Upgrade (+5) ' ] * 6 + [ ' Bomb Upgrade (+5) ' ] * 6 + [ ' Arrow Upgrade (+10) ' , ' Bomb Upgrade (+10) ' ] +
2017-06-24 18:48:03 +02:00
[ ' Progressive Armor ' ] * 2 + [ ' Progressive Shield ' ] * 3 + [ ' Progressive Glove ' ] * 2 +
2017-06-04 14:44:23 +02:00
[ ' Bottle ' ] * 4 +
[ ' Bombos ' , ' Book of Mudora ' , ' Blue Boomerang ' , ' Bow ' , ' Bug Catching Net ' , ' Cane of Byrna ' , ' Cane of Somaria ' ,
' Ether ' , ' Fire Rod ' , ' Flippers ' , ' Ocarina ' , ' Hammer ' , ' Hookshot ' , ' Ice Rod ' , ' Lamp ' , ' Cape ' , ' Magic Powder ' ,
' Red Boomerang ' , ' Mushroom ' , ' Pegasus Boots ' , ' Quake ' , ' Shovel ' , ' Silver Arrows ' ] +
[ ' Single Arrow ' , ' Sanctuary Heart Container ' , ' Rupees (100) ' ] + [ ' Boss Heart Container ' ] * 10 + [ ' Piece of Heart ' ] * 24 +
2017-08-01 17:25:08 +02:00
[ ' Rupees (50) ' ] * 7 + [ ' Rupees (5) ' ] * 4 + [ ' Rupee (1) ' ] * 2 + [ ' Rupees (300) ' ] * 5 + [ ' Rupees (20) ' ] * 28 +
[ ' Arrows (10) ' ] * 5 + [ ' Bombs (3) ' ] * 10 )
2017-05-15 20:28:04 +02:00
2017-06-24 18:48:03 +02:00
if world . mode == ' swordless ' :
world . push_item ( ' Ether Tablet ' , ItemFactory ( ' Rupees (20) ' ) , False )
world . push_item ( ' Bombos Tablet ' , ItemFactory ( ' Rupees (20) ' ) , False )
world . itempool . extend ( ItemFactory ( [ ' Rupees (20) ' , ' Rupees (20) ' ] ) )
elif world . mode == ' standard ' :
2017-06-24 11:11:56 +02:00
world . push_item ( ' Uncle ' , ItemFactory ( ' Progressive Sword ' ) , False )
world . get_location ( ' Uncle ' ) . event = True
2017-06-24 18:48:03 +02:00
world . itempool . extend ( ItemFactory ( [ ' Progressive Sword ' ] * 3 ) )
2017-05-16 21:23:47 +02:00
else :
2017-06-24 18:48:03 +02:00
world . itempool . extend ( ItemFactory ( [ ' Progressive Sword ' ] * 4 ) )
2017-05-16 21:23:47 +02:00
2017-05-22 19:52:50 +02:00
# provide mirror and pearl so you can avoid fake DW/LW and do dark world exploration as intended by algorithm, for now
if world . shuffle == ' insanity ' :
2017-06-24 11:11:56 +02:00
world . push_item ( ' [cave-040] Links House ' , ItemFactory ( ' Magic Mirror ' ) , False )
world . get_location ( ' [cave-040] Links House ' ) . event = True
world . push_item ( ' [dungeon-C-1F] Sanctuary ' , ItemFactory ( ' Moon Pearl ' ) , False )
world . get_location ( ' [dungeon-C-1F] Sanctuary ' ) . event = True
2017-05-22 19:52:50 +02:00
else :
2017-05-25 15:58:35 +02:00
world . itempool . extend ( ItemFactory ( [ ' Magic Mirror ' , ' Moon Pearl ' ] ) )
2017-05-22 19:52:50 +02:00
2017-05-16 21:23:47 +02:00
if world . goal == ' pedestal ' :
2017-06-23 21:32:31 +02:00
world . push_item ( ' Altar ' , ItemFactory ( ' Triforce ' ) , False )
2017-06-04 14:44:23 +02:00
elif world . goal == ' starhunt ' :
world . treasure_hunt_count = 10
world . treasure_hunt_icon = ' Power Star '
world . itempool . extend ( ItemFactory ( [ ' Power Star ' ] * 15 ) )
elif world . goal == ' triforcehunt ' :
world . treasure_hunt_count = 3
world . treasure_hunt_icon = ' Triforce Piece '
world . itempool . extend ( ItemFactory ( [ ' Triforce Piece ' ] * 3 ) )
2017-05-16 21:23:47 +02:00
2017-05-15 20:28:04 +02:00
if random . randint ( 0 , 3 ) == 0 :
2017-05-25 15:58:35 +02:00
world . itempool . append ( ItemFactory ( ' Magic Upgrade (1/4) ' ) )
2017-05-15 20:28:04 +02:00
else :
2017-05-25 15:58:35 +02:00
world . itempool . append ( ItemFactory ( ' Magic Upgrade (1/2) ' ) )
2017-05-15 20:28:04 +02:00
# distribute crystals
2017-07-17 22:29:32 +02:00
crystals = ItemFactory ( [ ' Green Pendant ' , ' Red Pendant ' , ' Blue Pendant ' ] )
crystal_locations = [ world . get_location ( ' Trinexx - Crystal ' ) ]
2017-05-15 20:28:04 +02:00
random . shuffle ( crystals )
2017-07-17 22:29:32 +02:00
if world . shuffle_ganon :
# ensure that no crystal gets locked inside of ganons tower location as that is unsolvable
for region , crystallocation in [ ( ' Eastern Palace ' , ' Armos - Pendant ' ) , ( ' Desert Palace North ' , ' Lanmolas - Pendant ' ) , ( ' Tower of Hera (Bottom) ' , ' Moldorm - Pendant ' ) ,
( ' Dark Palace (Entrance) ' , ' Helmasaur - Crystal ' ) , ( ' Thieves Town (Entrance) ' , ' Blind - Crystal ' ) , ( ' Skull Woods Final Section (Entrance) ' , ' Mothula - Crystal ' ) ,
( ' Swamp Palace (Entrance) ' , ' Arrghus - Crystal ' ) , ( ' Ice Palace (Entrance) ' , ' Kholdstare - Crystal ' ) , ( ' Misery Mire (Entrance) ' , ' Vitreous - Crystal ' ) ] :
if world . get_entrance ( ' Ganons Tower ' ) . connected_region . name == region :
# can't place a crystal here
world . push_item ( world . get_location ( crystallocation ) , crystals . pop ( ) , False )
2017-07-20 19:12:55 +02:00
world . get_location ( crystallocation ) . event = True
2017-07-17 22:29:32 +02:00
else :
crystal_locations . append ( world . get_location ( crystallocation ) )
else :
crystal_locations + = [ world . get_location ( ' Armos - Pendant ' ) , world . get_location ( ' Lanmolas - Pendant ' ) , world . get_location ( ' Moldorm - Pendant ' ) , world . get_location ( ' Helmasaur - Crystal ' ) ,
world . get_location ( ' Blind - Crystal ' ) , world . get_location ( ' Mothula - Crystal ' ) , world . get_location ( ' Arrghus - Crystal ' ) , world . get_location ( ' Kholdstare - Crystal ' ) ,
world . get_location ( ' Vitreous - Crystal ' ) ]
crystals . extend ( ItemFactory ( [ ' Crystal 1 ' , ' Crystal 2 ' , ' Crystal 3 ' , ' Crystal 4 ' , ' Crystal 7 ' ] ) )
random . shuffle ( crystals )
# check if dam is behind pyramid fairy, if so, swamp can't hold a crystal
if world . get_entrance ( ' Pyramid Fairy ' ) . connected_region . name == ' Dam ' :
2017-07-20 19:12:55 +02:00
try :
crystallocation = crystal_locations . pop ( crystal_locations . index ( world . get_location ( ' Arrghus - Crystal ' ) ) )
world . push_item ( world . get_location ( crystallocation ) , crystals . pop ( ) , False )
crystallocation . event = True
except ValueError :
pass
2017-07-17 22:29:32 +02:00
crystals . extend ( ItemFactory ( [ ' Crystal 5 ' , ' Crystal 6 ' ] ) )
random . shuffle ( crystals )
2017-05-15 20:28:04 +02:00
for location , crystal in zip ( crystal_locations , crystals ) :
world . push_item ( location , crystal , False )
2017-06-17 14:40:37 +02:00
location . event = True
2017-05-15 20:28:04 +02:00
# shuffle medallions
mm_medallion = [ ' Ether ' , ' Quake ' , ' Bombos ' ] [ random . randint ( 0 , 2 ) ]
tr_medallion = [ ' Ether ' , ' Quake ' , ' Bombos ' ] [ random . randint ( 0 , 2 ) ]
world . required_medallions = ( mm_medallion , tr_medallion )
2017-05-16 21:23:47 +02:00
def copy_world ( world ) :
# ToDo: Not good yet
2017-07-17 22:28:29 +02:00
ret = World ( world . shuffle , world . logic , world . mode , world . difficulty , world . goal , world . algorithm , world . place_dungeon_items , world . check_beatable_only , world . shuffle_ganon )
2017-05-16 21:23:47 +02:00
ret . required_medallions = list ( world . required_medallions )
2017-05-26 09:52:38 +02:00
ret . swamp_patch_required = world . swamp_patch_required
2017-06-04 13:10:22 +02:00
ret . treasure_hunt_count = world . treasure_hunt_count
ret . treasure_hunt_icon = world . treasure_hunt_icon
ret . sewer_light_cone = world . sewer_light_cone
ret . light_world_light_cone = world . light_world_light_cone
ret . dark_world_light_cone = world . dark_world_light_cone
2017-06-19 21:31:08 +02:00
ret . seed = world . seed
2017-07-17 23:13:39 +02:00
ret . can_access_trock_eyebridge = world . can_access_trock_eyebridge
2017-05-16 21:23:47 +02:00
create_regions ( ret )
# connect copied world
for region in world . regions :
for entrance in region . entrances :
ret . get_entrance ( entrance . name ) . connect ( ret . get_region ( region . name ) )
# fill locations
for location in world . get_locations ( ) :
if location . item is not None :
2017-06-04 14:43:13 +02:00
item = Item ( location . item . name , location . item . advancement , location . item . priority , location . item . key )
2017-05-16 21:23:47 +02:00
ret . get_location ( location . name ) . item = item
item . location = ret . get_location ( location . name )
2017-06-17 14:40:37 +02:00
if location . event :
ret . get_location ( location . name ) . event = True
2017-05-16 21:23:47 +02:00
# copy remaining itempool. No item in itempool should have an assigned location
for item in world . itempool :
2017-06-04 14:43:13 +02:00
ret . itempool . append ( Item ( item . name , item . advancement , item . priority , item . key ) )
2017-05-16 21:23:47 +02:00
# copy progress items in state
ret . state . prog_items = list ( world . state . prog_items )
2017-07-17 22:29:32 +02:00
set_rules ( ret )
2017-05-16 21:23:47 +02:00
return ret
def create_playthrough ( world ) :
# create a copy as we will modify it
2017-06-04 16:15:59 +02:00
old_world = world
2017-05-16 21:23:47 +02:00
world = copy_world ( world )
2017-05-26 09:53:34 +02:00
2017-06-04 14:44:23 +02:00
# in treasure hunt and pedestal goals, ganon is invincible
if world . goal in [ ' pedestal ' , ' starhunt ' , ' triforcehunt ' ] :
2017-05-26 09:53:34 +02:00
world . get_location ( ' Ganon ' ) . item = None
2017-05-16 21:23:47 +02:00
2017-06-23 22:15:29 +02:00
# if we only check for beatable, we can do this sanity check first before writing down spheres
if world . check_beatable_only and not world . can_beat_game ( ) :
raise RuntimeError ( ' Cannot beat game. Something went terribly wrong here! ' )
2017-05-16 21:23:47 +02:00
# get locations containing progress items
prog_locations = [ location for location in world . get_locations ( ) if location . item is not None and location . item . advancement ]
collection_spheres = [ ]
state = CollectionState ( world )
sphere_candidates = list ( prog_locations )
2017-05-26 09:55:24 +02:00
logging . getLogger ( ' ' ) . debug ( ' Building up collection spheres. ' )
2017-05-16 21:23:47 +02:00
while sphere_candidates :
2017-06-24 11:11:56 +02:00
state . sweep_for_events ( key_only = True )
2017-05-16 21:23:47 +02:00
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 sphere_candidates :
if state . can_reach ( location ) :
sphere . append ( location )
for location in sphere :
sphere_candidates . remove ( location )
2017-06-17 14:40:37 +02:00
state . collect ( location . item , True )
2017-05-16 21:23:47 +02:00
collection_spheres . append ( sphere )
2017-05-26 09:55:24 +02:00
logging . getLogger ( ' ' ) . debug ( ' Calculated sphere %i , containing %i of %i progress items. ' % ( len ( collection_spheres ) , len ( sphere ) , len ( prog_locations ) ) )
if not sphere :
2017-06-23 22:15:29 +02:00
logging . getLogger ( ' ' ) . debug ( ' The following items could not be reached: %s ' % [ ' %s at %s ' % ( location . item . name , location . name ) for location in sphere_candidates ] )
if not world . check_beatable_only :
raise RuntimeError ( ' Not all progression items reachable. Something went terribly wrong here. ' )
else :
break
2017-05-26 09:55:24 +02:00
2017-05-16 21:23:47 +02:00
# in the second phase, we cull each sphere such that the game is still beatable, reducing each range of influence to the bare minimum required inside it
for sphere in reversed ( collection_spheres ) :
to_delete = [ ]
for location in sphere :
# we remove the item at location and check if game is still beatable
2017-05-26 09:55:24 +02:00
logging . getLogger ( ' ' ) . debug ( ' Checking if %s is required to beat the game. ' % location . item . name )
2017-05-16 21:23:47 +02:00
old_item = location . item
location . item = None
state . remove ( old_item )
world . _item_cache = { } # need to invalidate
if world . can_beat_game ( ) :
to_delete . append ( location )
else :
# still required, got to keep it around
location . item = old_item
# cull entries in spheres for spoiler walkthrough at end
for location in to_delete :
sphere . remove ( location )
# we are now down to just the required progress items in collection_spheres in a minimum number of spheres. As a cleanup, we right trim empty spheres (can happen if we have multiple triforces)
collection_spheres = [ sphere for sphere in collection_spheres if sphere ]
2017-06-04 16:15:59 +02:00
# store the required locations for statistical analysis
old_world . required_locations = [ location . name for sphere in collection_spheres for location in sphere ]
2017-05-16 21:23:47 +02:00
# we can finally output our playthrough
2017-07-18 12:44:13 +02:00
old_world . spoiler . playthrough = OrderedDict ( [ ( str ( i + 1 ) , { str ( location ) : str ( location . item ) for location in sphere } ) for i , sphere in enumerate ( collection_spheres ) ] )
2017-05-15 20:28:04 +02:00