Ocarina of Time (#64)
* first commit (not including OoT data files yet)
* added some basic options
* rule parser works now at least
* make sure to commit everything this time
* temporary change to BaseClasses for oot
* overworld location graph builds mostly correctly
* adding oot data files
* commenting out world options until later since they only existed to make the RuleParser work
* conversion functions between AP ids and OOT ids
* world graph outputs
* set scrub prices
* itempool generates, entrances connected, way too many options added
* fixed set_rules and set_shop_rules
* temp baseclasses changes
* Reaches the fill step now, old event-based system retained in case the new way breaks
* Song placements and misc fixes everywhere
* temporary changes to make oot work
* changed root exits for AP fill framework
* prevent infinite recursion due to OoT sharing usage of the address field
* age reachability works hopefully, songs are broken again
* working spoiler log generation on beatable-only
* Logic tricks implemented
* need this for logic tricks
* fixed map/compass being placed on Serenade location
* kill unreachable events before filling the world
* add a bunch of utility functions to prepare for rom patching
* move OptionList into generic options
* fixed some silly bugs with OptionList
* properly seed all random behavior (so far)
* ROM generation working
* fix hints trying to get alttp dungeon hint texts
* continue fixing hints
* add oot to network data package
* change item and location IDs to 66000 and 67000 range respectively
* push removed items to precollected items
* fixed various issues with cross-contamination with multiple world generation
* reenable glitched logic (hopefully)
* glitched world files age-check fix
* cleaned up some get_locations calls
* added token shuffle and scrub shuffle, modified some options slightly to make the parsing work
* reenable MQ dungeons
* fix forest mq exception
* made targeting style an option for now, will be cosmetic later
* reminder to move targeting to cosmetics
* some oot option maintenance
* enabled starting time of day
* fixed issue breaking shop slots in multiworld generation
* added "off" option for text shuffle and hints
* shopsanity functionality restored
* change patch file extension
* remove unnecessary utility functions + imports
* update MIT license
* change option to "patch_uncompressed_rom" instead of "compress_rom"
* compliance with new AutoWorld systems
* Kill only internal events, remove non-internal big poe event in code
* re-add the big poe event and handle it correctly
* remove extra method in Range option
* fix typo
* Starting items, starting with consumables option
* do not remove nonexistent item
* move set_shop_rules to after shop items are placed
* some cleanup
* add retries for song placement
* flagged Skull Mask and Mask of Truth as advancement items
* update OoT to use LogicMixin
* Fixed trying to assign starting items from the wrong players
* fixed song retry step
* improved option handling, comments, and starting item replacements
* DefaultOnToggle writes Yes or No to spoiler
* enable compression of output if Compress executable is present
* clean up compression
* check whether (de)compressor exists before running the process
* allow specification of rom path in host.yaml
* check if decompressed file already exists before decompressing again
* fix triforce hunt generation
* rename all the oot state functions with prefix
* OoT: mark triforce pieces as completion goal for triforce hunt
* added overworld and any-dungeon shuffle for dungeon items
* Hide most unshuffled locations and events from the list of locations in spoiler
* build oot option ranges with a generic function instead of defining each separately
* move oot output-type control to host.yaml instead of individual yamls
* implement dungeon song shuffle
* minor improvements to overworld dungeon item shuffle
* remove random ice trap names in shops, mostly to avoid maintaining a massive censor list
* always output patch file to folder, remove option to generate ROM in preparation for removal
* re-add the fix for infinite recursion due to not being light or dark world
* change AP-sendable to Ocarina of Time model, since the triforce piece has some extra code apparently
* oot: remove item_names and location_names
* oot: minor fixes
* oot: comment out ROM patching
* oot: only add CollectionState objects on creation if actually needed
* main entrance shuffle method and entrances-based rules
* fix entrances based rules
* disable master quest and big poe count options for client compatibility
* use get_player_name instead of get_player_names
* fix OptionList
* fix oot options for new option system
* new coop section in oot rom: expand player names to 16 bytes, write AP_PLAYER_NAME at end of PLAYER_NAMES
* fill AP player name in oot rom with 0 instead of 0xDF
* encode player name with ASCII for fixed-width
* revert oot player name array to 8 bytes per name
* remove Pierre location if fast scarecrow is on
* check player name length
* "free_scarecrow" not "fast_scarecrow"
* OoT locations now properly store the AP ID instead of the oot internal ID
* oot __version__ updates in lockstep with AP version
* pull in unmodified oot cosmetic files
* also grab JSONDump since it's needed apparently
* gather extra needed methods, modify imports
* delete cosmetics log, replace all instances of SettingsList with OOTWorld
* cosmetic options working, except for sound effects (due to ear-safe issues)
* SFX, Music, and Fanfare randomization reenabled
* move OoT data files into the worlds folder
* move Compress and Decompress into oot data folder
* Replace get_all_state with custom method to avoid the cache
* OoT ROM: increment item counter before setting incoming item/player values to 0, preventing desync issues
* set data_version to 0
* make Kokiri Sword shuffle off by default
* reenable "Random Choice" for various cosmetic options
* kill Ruto's Letter turnin if open fountain
also fix for shopsanity
* place Buy Goron/Zora Tunic first in shop shuffle
* make ice traps appear as other items instead of breaking generation
* managed to break ice traps on non-major-only
* only handle ice traps if they are on
* fix shopsanity for non-oot games, and write player name instead of player number
* light arrows hint uses player name instead of player number
* Reenable "skip child zelda" option
* fix entrances_based_rules
* fix ganondorf hint if starting with light arrows
* fix dungeonitem shuffle and shopsanity interaction
* remove has_all_of, has_any_of, count_of in BaseClasses, replace usage with has_all, has_any, has_group
* force local giveable item on ZL if skip_child_zelda and shuffle_song_items is any
* keep bosses and bombchu bowling chus out of data package
* revert workaround for infinite recursion and fix it properly
* fix shared shop id caches during patching process
* fix shop text box overflows, as much as possible
* add default oot host.yaml option
* add .apz5, .n64, .z64 to gitignore
* Properly document and name all (functioning) OOT options
* clean up some imports
* remove unnecessary files from oot's data
* fix typo in gitignore
* readd the Compress and Decompress utilities, since they are needed for generation
* cleanup of imports and some minor optimizations
* increase shop offset for item IDs to 0xCB
* remove shop item AP ids entirely
* prevent triforce pieces for other players from being received by yourself
* add "excluded" property to Location
* Hint system adapted and reenabled; hints still unseeded
* make hints deterministic with lists instead of sets
* do not allow hints to point to Light Arrows on non-vanilla bridge
* foreign locations hint as their full name in OoT rather than their region
* checkedLocations now stores hint names by player ID, so that the same location in different worlds can have hints associated
* consolidate versioning in Utils
* ice traps appear as major items rather than any progression item
* set prescription and claim check as defaults for adult trade item settings
* add oot options to playerSettings
* allow case-insensitive logic tricks in yaml
* fix oot shopsanity option formatting
* Write OoT override info even if local item, enabling local checks to show up immediately in the client
* implement CollectionState.can_live_dmg for oot glitched logic
* filter item names for invalid characters when patching shops
* make ice traps appear according to the settings of the world they are shuffled into, rather than the original world
* set hidden-spoiler items and locations with Shop items to events
* make GF carpenters, Gerudo Card, Malon, ZL, and Impa events if the relevant settings are enabled, preventing them from appearing in the client on game start
* Fix oot Glitched and No Logic generation
* fix indenting
* Greatly reduce displayed cosmetic options
* Change oot data version to 1
* add apz5 distribution to webhost
* print player name if an ALttP dungeon contains a good item for OoT world
* delete unneeded commented code
* remove OcarinaSongs import to satisfy lint
											 
										 
										
											2021-09-02 08:35:05 -04:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								# text details: https://wiki.cloudmodding.com/oot/Text_Format  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								import  random  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  . TextBox  import  line_wrap  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								TEXT_START  =  0x92D000  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								ENG_TEXT_SIZE_LIMIT  =  0x39000  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								JPN_TEXT_SIZE_LIMIT  =  0x3A150  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								JPN_TABLE_START  =  0xB808AC  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								ENG_TABLE_START  =  0xB849EC  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								CREDITS_TABLE_START  =  0xB88C0C  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								JPN_TABLE_SIZE  =  ENG_TABLE_START  -  JPN_TABLE_START  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								ENG_TABLE_SIZE  =  CREDITS_TABLE_START  -  ENG_TABLE_START  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								EXTENDED_TABLE_START  =  JPN_TABLE_START  # start writing entries to the jp table instead of english for more space  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								EXTENDED_TABLE_SIZE  =  JPN_TABLE_SIZE  +  ENG_TABLE_SIZE  # 0x8360 bytes, 4204 entries  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# name of type, followed by number of additional bytes to read, follwed by a function that prints the code  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								CONTROL_CODES  =  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00 :  ( ' pad ' ,  0 ,  lambda  _ :  ' <pad> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x01 :  ( ' line-break ' ,  0 ,  lambda  _ :  ' \n '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x02 :  ( ' end ' ,  0 ,  lambda  _ :  ' '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x04 :  ( ' box-break ' ,  0 ,  lambda  _ :  ' \n ▼ \n '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x05 :  ( ' color ' ,  1 ,  lambda  d :  ' <color  '  +  " {:02x} " . format ( d )  +  ' > '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x06 :  ( ' gap ' ,  1 ,  lambda  d :  ' < '  +  str ( d )  +  ' px gap> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x07 :  ( ' goto ' ,  2 ,  lambda  d :  ' <goto  '  +  " {:04x} " . format ( d )  +  ' > '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x08 :  ( ' instant ' ,  0 ,  lambda  _ :  ' <allow instant text> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x09 :  ( ' un-instant ' ,  0 ,  lambda  _ :  ' <disallow instant text> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0A :  ( ' keep-open ' ,  0 ,  lambda  _ :  ' <keep open> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0B :  ( ' event ' ,  0 ,  lambda  _ :  ' <event> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0C :  ( ' box-break-delay ' ,  1 ,  lambda  d :  ' \n ▼<wait  '  +  str ( d )  +  '  frames> \n '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0E :  ( ' fade-out ' ,  1 ,  lambda  d :  ' <fade after  '  +  str ( d )  +  '  frames?> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0F :  ( ' name ' ,  0 ,  lambda  _ :  ' <name> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x10 :  ( ' ocarina ' ,  0 ,  lambda  _ :  ' <ocarina> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x12 :  ( ' sound ' ,  2 ,  lambda  d :  ' <play SFX  '  +  " {:04x} " . format ( d )  +  ' > '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x13 :  ( ' icon ' ,  1 ,  lambda  d :  ' <icon  '  +  " {:02x} " . format ( d )  +  ' > '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x14 :  ( ' speed ' ,  1 ,  lambda  d :  ' <delay each character by  '  +  str ( d )  +  '  frames> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x15 :  ( ' background ' ,  3 ,  lambda  d :  ' <set background to  '  +  " {:06x} " . format ( d )  +  ' > '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x16 :  ( ' marathon ' ,  0 ,  lambda  _ :  ' <marathon time> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x17 :  ( ' race ' ,  0 ,  lambda  _ :  ' <race time> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x18 :  ( ' points ' ,  0 ,  lambda  _ :  ' <points> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x19 :  ( ' skulltula ' ,  0 ,  lambda  _ :  ' <skulltula count> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x1A :  ( ' unskippable ' ,  0 ,  lambda  _ :  ' <text is unskippable> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x1B :  ( ' two-choice ' ,  0 ,  lambda  _ :  ' <start two choice> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x1C :  ( ' three-choice ' ,  0 ,  lambda  _ :  ' <start three choice> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x1D :  ( ' fish ' ,  0 ,  lambda  _ :  ' <fish weight> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x1E :  ( ' high-score ' ,  1 ,  lambda  d :  ' <high-score  '  +  " {:02x} " . format ( d )  +  ' > '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x1F :  ( ' time ' ,  0 ,  lambda  _ :  ' <current time> '  ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								SPECIAL_CHARACTERS  =  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x80 :  ' À ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x81 :  ' Á ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x82 :  ' Â ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x83 :  ' Ä ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x84 :  ' Ç ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x85 :  ' È ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x86 :  ' É ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x87 :  ' Ê ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x88 :  ' Ë ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x89 :  ' Ï ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x8A :  ' Ô ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x8B :  ' Ö ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x8C :  ' Ù ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x8D :  ' Û ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x8E :  ' Ü ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x8F :  ' ß ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x90 :  ' à ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x91 :  ' á ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x92 :  ' â ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x93 :  ' ä ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x94 :  ' ç ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x95 :  ' è ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x96 :  ' é ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x97 :  ' ê ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x98 :  ' ë ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x99 :  ' ï ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9A :  ' ô ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9B :  ' ö ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9C :  ' ù ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9D :  ' û ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9E :  ' ü ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9F :  ' [A] ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0xA0 :  ' [B] ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0xA1 :  ' [C] ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0xA2 :  ' [L] ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0xA3 :  ' [R] ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0xA4 :  ' [Z] ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0xA5 :  ' [C Up] ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0xA6 :  ' [C Down] ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0xA7 :  ' [C Left] ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0xA8 :  ' [C Right] ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0xA9 :  ' [Triangle] ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0xAA :  ' [Control Stick] ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								UTF8_TO_OOT_SPECIAL  =  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x80 ) :  0x80 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xae ) :  0x81 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x82 ) :  0x82 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x84 ) :  0x83 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x87 ) :  0x84 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x88 ) :  0x85 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x89 ) :  0x86 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x8a ) :  0x87 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x8b ) :  0x88 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x8f ) :  0x89 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x94 ) :  0x8A , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x96 ) :  0x8B , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x99 ) :  0x8C , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x9b ) :  0x8D , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x9c ) :  0x8E , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0x9f ) :  0x8F , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xa0 ) :  0x90 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xa1 ) :  0x91 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xa2 ) :  0x92 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xa4 ) :  0x93 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xa7 ) :  0x94 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xa8 ) :  0x95 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xa9 ) :  0x96 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xaa ) :  0x97 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xab ) :  0x98 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xaf ) :  0x99 , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xb4 ) :  0x9A , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xb6 ) :  0x9B , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xb9 ) :  0x9C , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xbb ) :  0x9D , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ( 0xc3 ,  0xbc ) :  0x9E , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								GOSSIP_STONE_MESSAGES  =  list (  range ( 0x0401 ,  0x04FF )  )  # ids of the actual hints  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								GOSSIP_STONE_MESSAGES  + =  [ 0x2053 ,  0x2054 ]  # shared initial stone messages  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								TEMPLE_HINTS_MESSAGES  =  [ 0x7057 ,  0x707A ]  # dungeon reward hints from the temple of time pedestal  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								LIGHT_ARROW_HINT  =  [ 0x70CC ]  # ganondorf's light arrow hint line  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								GS_TOKEN_MESSAGES  =  [ 0x00B4 ,  0x00B5 ]  # Get Gold Skulltula Token messages  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								ERROR_MESSAGE  =  0x0001  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# messages for shorter item messages  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# ids are in the space freed up by move_shop_item_messages()  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								ITEM_MESSAGES  =  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0001 :  " \x08 \x06 \x30 \x05 \x41 TEXT ID ERROR! \x05 \x40 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9001 :  " \x08 \x13 \x2D You borrowed a  \x05 \x41 Pocket Egg \x05 \x40 ! \x01 A Pocket Cucco will hatch from \x01 it overnight. Be sure to give it \x01 back. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0002 :  " \x08 \x13 \x2F You returned the Pocket Cucco \x01 and got  \x05 \x41 Cojiro \x05 \x40  in return! \x01 Unlike other Cuccos, Cojiro \x01 rarely crows. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0003 :  " \x08 \x13 \x30 You got an  \x05 \x41 Odd Mushroom \x05 \x40 ! \x01 It is sure to spoil quickly! Take \x01 it to the Kakariko Potion Shop. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0004 :  " \x08 \x13 \x31 You received an  \x05 \x41 Odd Potion \x05 \x40 ! \x01 It may be useful for something... \x01 Hurry to the Lost Woods! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0005 :  " \x08 \x13 \x32 You returned the Odd Potion  \x01 and got the  \x05 \x41 Poacher ' s Saw \x05 \x40 ! \x01 The young punk guy must have \x01 left this. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0007 :  " \x08 \x13 \x48 You got a  \x01 \x05 \x41 Deku Seeds Bullet Bag \x05 \x40 . \x01 This bag can hold up to  \x05 \x46 40 \x05 \x40 \x01 slingshot bullets. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0008 :  " \x08 \x13 \x33 You traded the Poacher ' s Saw  \x01 for a  \x05 \x41 Broken Goron ' s Sword \x05 \x40 ! \x01 Visit Biggoron to get it repaired! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0009 :  " \x08 \x13 \x34 You checked in the Broken  \x01 Goron ' s Sword and received a  \x01 \x05 \x41 Prescription \x05 \x40 ! \x01 Go see King Zora! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x000A :  " \x08 \x13 \x37 The Biggoron ' s Sword... \x01 You got a  \x05 \x41 Claim Check  \x05 \x40 for it! \x01 You can ' t wait for the sword! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x000B :  " \x08 \x13 \x2E You got a  \x05 \x41 Pocket Cucco,  \x05 \x40 one \x01 of Anju ' s prized hens! It fits  \x01 in your pocket. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x000C :  " \x08 \x13 \x3D You got the  \x05 \x41 Biggoron ' s Sword \x05 \x40 ! \x01 This blade was forged by a  \x01 master smith and won ' t break! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x000D :  " \x08 \x13 \x35 You used the Prescription and \x01 received an  \x05 \x41 Eyeball Frog \x05 \x40 ! \x01 Be quick and deliver it to Lake  \x01 Hylia! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x000E :  " \x08 \x13 \x36 You traded the Eyeball Frog  \x01 for the  \x05 \x41 World ' s Finest Eye Drops \x05 \x40 ! \x01 Hurry! Take them to Biggoron! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0010 :  " \x08 \x13 \x25 You borrowed a  \x05 \x41 Skull Mask \x05 \x40 . \x01 You feel like a monster while you \x01 wear this mask! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0011 :  " \x08 \x13 \x26 You borrowed a  \x05 \x41 Spooky Mask \x05 \x40 . \x01 You can scare many people \x01 with this mask! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0012 :  " \x08 \x13 \x24 You borrowed a  \x05 \x41 Keaton Mask \x05 \x40 . \x01 You ' ll be a popular guy with \x01 this mask on! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0013 :  " \x08 \x13 \x27 You borrowed a  \x05 \x41 Bunny Hood \x05 \x40 . \x01 The hood ' s long ears are so \x01 cute! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0014 :  " \x08 \x13 \x28 You borrowed a  \x05 \x41 Goron Mask \x05 \x40 . \x01 It will make your head look \x01 big, though. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0015 :  " \x08 \x13 \x29 You borrowed a  \x05 \x41 Zora Mask \x05 \x40 . \x01 With this mask, you can \x01 become one of the Zoras! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0016 :  " \x08 \x13 \x2A You borrowed a  \x05 \x41 Gerudo Mask \x05 \x40 . \x01 This mask will make you look \x01 like...a girl? " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0017 :  " \x08 \x13 \x2B You borrowed a  \x05 \x41 Mask of Truth \x05 \x40 . \x01 Show it to many people! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0030 :  " \x08 \x13 \x06 You found the  \x05 \x41 Fairy Slingshot \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0031 :  " \x08 \x13 \x03 You found the  \x05 \x41 Fairy Bow \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0032 :  " \x08 \x13 \x02 You got  \x05 \x41 Bombs \x05 \x40 ! \x01 If you see something \x01 suspicious, bomb it! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0033 :  " \x08 \x13 \x09 You got  \x05 \x41 Bombchus \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0034 :  " \x08 \x13 \x01 You got a  \x05 \x41 Deku Nut \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0035 :  " \x08 \x13 \x0E You found the  \x05 \x41 Boomerang \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0036 :  " \x08 \x13 \x0A You found the  \x05 \x41 Hookshot \x05 \x40 ! \x01 It ' s a spring-loaded chain that \x01 you can cast out to hook things. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0037 :  " \x08 \x13 \x00 You got a  \x05 \x41 Deku Stick \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0038 :  " \x08 \x13 \x11 You found the  \x05 \x41 Megaton Hammer \x05 \x40 ! \x01 It ' s so heavy, you need to \x01 use two hands to swing it! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0039 :  " \x08 \x13 \x0F You found the  \x05 \x41 Lens of Truth \x05 \x40 ! \x01 Mysterious things are hidden \x01 everywhere! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x003A :  " \x08 \x13 \x08 You found the  \x05 \x41 Ocarina of Time \x05 \x40 ! \x01 It glows with a mystical light... " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x003C :  " \x08 \x13 \x67 You received the  \x05 \x41 Fire \x01 Medallion \x05 \x40 ! \x01 Darunia awakens as a Sage and \x01 adds his power to yours! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x003D :  " \x08 \x13 \x68 You received the  \x05 \x43 Water \x01 Medallion \x05 \x40 ! \x01 Ruto awakens as a Sage and \x01 adds her power to yours! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x003E :  " \x08 \x13 \x66 You received the  \x05 \x42 Forest \x01 Medallion \x05 \x40 ! \x01 Saria awakens as a Sage and \x01 adds her power to yours! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x003F :  " \x08 \x13 \x69 You received the  \x05 \x46 Spirit \x01 Medallion \x05 \x40 ! \x01 Nabooru awakens as a Sage and \x01 adds her power to yours! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0040 :  " \x08 \x13 \x6B You received the  \x05 \x44 Light \x01 Medallion \x05 \x40 ! \x01 Rauru the Sage adds his power \x01 to yours! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0041 :  " \x08 \x13 \x6A You received the  \x05 \x45 Shadow \x01 Medallion \x05 \x40 ! \x01 Impa awakens as a Sage and \x01 adds her power to yours! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0042 :  " \x08 \x13 \x14 You got an  \x05 \x41 Empty Bottle \x05 \x40 ! \x01 You can put something in this \x01 bottle. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0043 :  " \x08 \x13 \x15 You got a  \x05 \x41 Red Potion \x05 \x40 ! \x01 It will restore your health " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0044 :  " \x08 \x13 \x16 You got a  \x05 \x42 Green Potion \x05 \x40 ! \x01 It will restore your magic. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0045 :  " \x08 \x13 \x17 You got a  \x05 \x43 Blue Potion \x05 \x40 ! \x01 It will recover your health \x01 and magic. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0046 :  " \x08 \x13 \x18 You caught a  \x05 \x41 Fairy \x05 \x40  in a bottle! \x01 It will revive you \x01 the moment you run out of life  \x01 energy. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0047 :  " \x08 \x13 \x19 You got a  \x05 \x41 Fish \x05 \x40 ! \x01 It looks so fresh and \x01 delicious! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0048 :  " \x08 \x13 \x10 You got a  \x05 \x41 Magic Bean \x05 \x40 ! \x01 Find a suitable spot for a garden \x01 and plant it. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9048 :  " \x08 \x13 \x10 You got a  \x05 \x41 Pack of Magic Beans \x05 \x40 ! \x01 Find suitable spots for a garden \x01 and plant them. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x004A :  " \x08 \x13 \x07 You received the  \x05 \x41 Fairy Ocarina \x05 \x40 ! \x01 This is a memento from Saria. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x004B :  " \x08 \x13 \x3D You got the  \x05 \x42 Giant ' s Knife \x05 \x40 ! \x01 Hold it with both hands to \x01 attack! It ' s so long, you \x01 can ' t use it with a  \x05 \x44 shield \x05 \x40 . " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x004C :  " \x08 \x13 \x3E You got a  \x05 \x44 Deku Shield \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x004D :  " \x08 \x13 \x3F You got a  \x05 \x44 Hylian Shield \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x004E :  " \x08 \x13 \x40 You found the  \x05 \x44 Mirror Shield \x05 \x40 ! \x01 The shield ' s polished surface can \x01 reflect light or energy. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x004F :  " \x08 \x13 \x0B You found the  \x05 \x41 Longshot \x05 \x40 ! \x01 It ' s an upgraded Hookshot. \x01 It extends  \x05 \x41 twice \x05 \x40  as far! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0050 :  " \x08 \x13 \x42 You got a  \x05 \x41 Goron Tunic \x05 \x40 ! \x01 Going to a hot place? No worry! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0051 :  " \x08 \x13 \x43 You got a  \x05 \x43 Zora Tunic \x05 \x40 ! \x01 Wear it, and you won ' t drown \x01 underwater. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0052 :  " \x08 You got a  \x05 \x42 Magic Jar \x05 \x40 ! \x01 Your Magic Meter is filled! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0053 :  " \x08 \x13 \x45 You got the  \x05 \x41 Iron Boots \x05 \x40 ! \x01 So heavy, you can ' t run. \x01 So heavy, you can ' t float. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0054 :  " \x08 \x13 \x46 You got the  \x05 \x41 Hover Boots \x05 \x40 ! \x01 With these mysterious boots \x01 you can hover above the ground. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0055 :  " \x08 You got a  \x05 \x45 Recovery Heart \x05 \x40 ! \x01 Your life energy is recovered! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0056 :  " \x08 \x13 \x4B You upgraded your quiver to a \x01 \x05 \x41 Big Quiver \x05 \x40 ! \x01 Now you can carry more arrows- \x01 \x05 \x46 40  \x05 \x40 in total! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0057 :  " \x08 \x13 \x4C You upgraded your quiver to \x01 the  \x05 \x41 Biggest Quiver \x05 \x40 ! \x01 Now you can carry to a \x01 maximum of  \x05 \x46 50 \x05 \x40  arrows! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0058 :  " \x08 \x13 \x4D You found a  \x05 \x41 Bomb Bag \x05 \x40 ! \x01 You found  \x05 \x41 20 Bombs \x05 \x40  inside! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0059 :  " \x08 \x13 \x4E You got a  \x05 \x41 Big Bomb Bag \x05 \x40 ! \x01 Now you can carry more  \x01 Bombs, up to a maximum of  \x05 \x46 30 \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x005A :  " \x08 \x13 \x4F You got the  \x01 \x05 \x41 Biggest Bomb Bag \x05 \x40 ! \x01 Now, you can carry up to  \x01 \x05 \x46 40 \x05 \x40  Bombs! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x005B :  " \x08 \x13 \x51 You found the  \x05 \x43 Silver Gauntlets \x05 \x40 ! \x01 You feel the power to lift \x01 big things with it! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x005C :  " \x08 \x13 \x52 You found the  \x05 \x43 Golden Gauntlets \x05 \x40 ! \x01 You can feel even more power \x01 coursing through your arms! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x005D :  " \x08 \x13 \x1C You put a  \x05 \x44 Blue Fire \x05 \x40 \x01 into the bottle! \x01 This is a cool flame you can \x01 use on red ice. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x005E :  " \x08 \x13 \x56 You got an  \x05 \x43 Adult ' s Wallet \x05 \x40 ! \x01 Now you can hold \x01 up to  \x05 \x46 200 \x05 \x40   \x05 \x46 Rupees \x05 \x40 . " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x005F :  " \x08 \x13 \x57 You got a  \x05 \x43 Giant ' s Wallet \x05 \x40 ! \x01 Now you can hold \x01 up to  \x05 \x46 500 \x05 \x40   \x05 \x46 Rupees \x05 \x40 . " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0060 :  " \x08 \x13 \x77 You found a  \x05 \x41 Small Key \x05 \x40 ! \x01 This key will open a locked  \x01 door. You can use it only \x01 in this dungeon. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0066 :  " \x08 \x13 \x76 You found the  \x05 \x41 Dungeon Map \x05 \x40 ! \x01 It ' s the map to this dungeon. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0067 :  " \x08 \x13 \x75 You found the  \x05 \x41 Compass \x05 \x40 ! \x01 Now you can see the locations \x01 of many hidden things in the \x01 dungeon! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0068 :  " \x08 \x13 \x6F You obtained the  \x05 \x41 Stone of Agony \x05 \x40 ! \x01 If you equip a  \x05 \x44 Rumble Pak \x05 \x40 , it \x01 will react to nearby...secrets. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0069 :  " \x08 \x13 \x23 You received  \x05 \x41 Zelda ' s Letter \x05 \x40 ! \x01 Wow! This letter has Princess \x01 Zelda ' s autograph! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x006C :  " \x08 \x13 \x49 Your  \x05 \x41 Deku Seeds Bullet Bag  \x01 \x05 \x40 has become bigger! \x01 This bag can hold  \x05 \x46 50 \x05 \x41   \x05 \x40 bullets! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x006F :  " \x08 You got a  \x05 \x42 Green Rupee \x05 \x40 ! \x01 That ' s  \x05 \x42 one Rupee \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0070 :  " \x08 \x13 \x04 You got the  \x05 \x41 Fire Arrow \x05 \x40 ! \x01 If you hit your target, \x01 it will catch fire. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0071 :  " \x08 \x13 \x0C You got the  \x05 \x43 Ice Arrow \x05 \x40 ! \x01 If you hit your target, \x01 it will freeze. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0072 :  " \x08 \x13 \x12 You got the  \x05 \x44 Light Arrow \x05 \x40 ! \x01 The light of justice \x01 will smite evil! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0073 :  " \x08 \x06 \x28 You have learned the \x01 \x06 \x2F \x05 \x42 Minuet of Forest \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0074 :  " \x08 \x06 \x28 You have learned the \x01 \x06 \x37 \x05 \x41 Bolero of Fire \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0075 :  " \x08 \x06 \x28 You have learned the \x01 \x06 \x29 \x05 \x43 Serenade of Water \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0076 :  " \x08 \x06 \x28 You have learned the \x01 \x06 \x2D \x05 \x46 Requiem of Spirit \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0077 :  " \x08 \x06 \x28 You have learned the \x01 \x06 \x28 \x05 \x45 Nocturne of Shadow \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0078 :  " \x08 \x06 \x28 You have learned the \x01 \x06 \x32 \x05 \x44 Prelude of Light \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0079 :  " \x08 \x13 \x50 You got the  \x05 \x41 Goron ' s Bracelet \x05 \x40 ! \x01 Now you can pull up Bomb \x01 Flowers. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x007A :  " \x08 \x13 \x1D You put a  \x05 \x41 Bug  \x05 \x40 in the bottle! \x01 This kind of bug prefers to \x01 live in small holes in the ground. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x007B :  " \x08 \x13 \x70 You obtained the  \x05 \x41 Gerudo ' s  \x01 Membership Card \x05 \x40 ! \x01 You can get into the Gerudo ' s \x01 training ground. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0080 :  " \x08 \x13 \x6C You got the  \x05 \x42 Kokiri ' s Emerald \x05 \x40 ! \x01 This is the Spiritual Stone of  \x01 Forest passed down by the \x01 Great Deku Tree. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0081 :  " \x08 \x13 \x6D You obtained the  \x05 \x41 Goron ' s Ruby \x05 \x40 ! \x01 This is the Spiritual Stone of  \x01 Fire passed down by the Gorons! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0082 :  " \x08 \x13 \x6E You obtained  \x05 \x43 Zora ' s Sapphire \x05 \x40 ! \x01 This is the Spiritual Stone of \x01 Water passed down by the \x01 Zoras! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0090 :  " \x08 \x13 \x00 Now you can pick up  \x01 many  \x05 \x41 Deku Sticks \x05 \x40 ! \x01 You can carry up to  \x05 \x46 20 \x05 \x40  of them! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0091 :  " \x08 \x13 \x00 You can now pick up  \x01 even more  \x05 \x41 Deku Sticks \x05 \x40 ! \x01 You can carry up to  \x05 \x46 30 \x05 \x40  of them! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0097 :  " \x08 \x13 \x20 You caught a  \x05 \x41 Poe  \x05 \x40 in a bottle! \x01 Something good might happen! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0098 :  " \x08 \x13 \x1A You got  \x05 \x41 Lon Lon Milk \x05 \x40 ! \x01 This milk is very nutritious! \x01 There are two drinks in it. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0099 :  " \x08 \x13 \x1B You found  \x05 \x41 Ruto ' s Letter \x05 \x40  in a \x01 bottle! Show it to King Zora. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9099 :  " \x08 \x13 \x1B You found  \x05 \x41 a letter in a bottle \x05 \x40 ! \x01 You remove the letter from the \x01 bottle, freeing it for other uses. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x009A :  " \x08 \x13 \x21 You got a  \x05 \x41 Weird Egg \x05 \x40 ! \x01 Feels like there ' s something \x01 moving inside! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00A4 :  " \x08 \x13 \x3B You got the  \x05 \x42 Kokiri Sword \x05 \x40 ! \x01 This is a hidden treasure of \x01 the Kokiri. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00A7 :  " \x08 \x13 \x01 Now you can carry \x01 many  \x05 \x41 Deku Nuts \x05 \x40 ! \x01 You can hold up to  \x05 \x46 30 \x05 \x40  nuts! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00A8 :  " \x08 \x13 \x01 You can now carry even \x01 more  \x05 \x41 Deku Nuts \x05 \x40 ! You can carry \x01 up to  \x05 \x46 40 \x05 \x41   \x05 \x40 nuts! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00AD :  " \x08 \x13 \x05 You got  \x05 \x41 Din ' s Fire \x05 \x40 ! \x01 Its fireball engulfs everything! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00AE :  " \x08 \x13 \x0D You got  \x05 \x42 Farore ' s Wind \x05 \x40 ! \x01 This is warp magic you can use! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00AF :  " \x08 \x13 \x13 You got  \x05 \x43 Nayru ' s Love \x05 \x40 ! \x01 Cast this to create a powerful \x01 protective barrier. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00B4 :  " \x08 You got a  \x05 \x41 Gold Skulltula Token \x05 \x40 ! \x01 You ' ve collected  \x05 \x41 \x19 \x05 \x40  tokens in total. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00B5 :  " \x08 You destroyed a  \x05 \x41 Gold Skulltula \x05 \x40 . \x01 You got a token proving you  \x01 destroyed it! " ,  #Unused 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00C2 :  " \x08 \x13 \x73 You got a  \x05 \x41 Piece of Heart \x05 \x40 ! \x01 Collect four pieces total to get \x01 another Heart Container. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00C3 :  " \x08 \x13 \x73 You got a  \x05 \x41 Piece of Heart \x05 \x40 ! \x01 So far, you ' ve collected two  \x01 pieces. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00C4 :  " \x08 \x13 \x73 You got a  \x05 \x41 Piece of Heart \x05 \x40 ! \x01 Now you ' ve collected three  \x01 pieces! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00C5 :  " \x08 \x13 \x73 You got a  \x05 \x41 Piece of Heart \x05 \x40 ! \x01 You ' ve completed another Heart \x01 Container! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00C6 :  " \x08 \x13 \x72 You got a  \x05 \x41 Heart Container \x05 \x40 ! \x01 Your maximum life energy is  \x01 increased by one heart. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00C7 :  " \x08 \x13 \x74 You got the  \x05 \x41 Boss Key \x05 \x40 ! \x01 Now you can get inside the  \x01 chamber where the Boss lurks. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9002 :  " \x08 You are a  \x05 \x43 FOOL \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00CC :  " \x08 You got a  \x05 \x43 Blue Rupee \x05 \x40 ! \x01 That ' s  \x05 \x43 five Rupees \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00CD :  " \x08 \x13 \x53 You got the  \x05 \x43 Silver Scale \x05 \x40 ! \x01 You can dive deeper than you \x01 could before. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00CE :  " \x08 \x13 \x54 You got the  \x05 \x43 Golden Scale \x05 \x40 ! \x01 Now you can dive much \x01 deeper than you could before! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00D1 :  " \x08 \x06 \x14 You ' ve learned  \x05 \x42 Saria ' s Song \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00D2 :  " \x08 \x06 \x11 You ' ve learned  \x05 \x41 Epona ' s Song \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00D3 :  " \x08 \x06 \x0B You ' ve learned the  \x05 \x46 Sun ' s Song \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00D4 :  " \x08 \x06 \x15 You ' ve learned  \x05 \x43 Zelda ' s Lullaby \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00D5 :  " \x08 \x06 \x05 You ' ve learned the  \x05 \x44 Song of Time \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00D6 :  " \x08 You ' ve learned the  \x05 \x45 Song of Storms \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00DC :  " \x08 \x13 \x58 You got  \x05 \x41 Deku Seeds \x05 \x40 ! \x01 Use these as bullets \x01 for your Slingshot. " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00DD :  " \x08 You mastered the secret sword \x01 technique of the  \x05 \x41 Spin Attack \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00E4 :  " \x08 You can now use  \x05 \x42 Magic \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00E5 :  " \x08 Your  \x05 \x44 defensive power \x05 \x40  is enhanced! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00E6 :  " \x08 You got a  \x05 \x46 bundle of arrows \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00E8 :  " \x08 Your magic power has been  \x01 enhanced! Now you have twice \x01 as much  \x05 \x41 Magic Power \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00E9 :  " \x08 Your defensive power has been  \x01 enhanced! Damage inflicted by  \x01 enemies will be  \x05 \x41 reduced by half \x05 \x40 . " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00F0 :  " \x08 You got a  \x05 \x41 Red Rupee \x05 \x40 ! \x01 That ' s  \x05 \x41 twenty Rupees \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00F1 :  " \x08 You got a  \x05 \x45 Purple Rupee \x05 \x40 ! \x01 That ' s  \x05 \x45 fifty Rupees \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00F2 :  " \x08 You got a  \x05 \x46 Huge Rupee \x05 \x40 ! \x01 This Rupee is worth a whopping \x01 \x05 \x46 two hundred Rupees \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00F9 :  " \x08 \x13 \x1E You put a  \x05 \x41 Big Poe  \x05 \x40 in a bottle! \x01 Let ' s sell it at the  \x05 \x41 Ghost Shop \x05 \x40 ! \x01 Something good might happen! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9003 :  " \x08 You found a piece of the  \x05 \x41 Triforce \x05 \x40 ! " , 
							 
						 
					
						
							
								
									
										
										
										
											2021-09-10 10:04:56 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    0x9097 :  " \x08 You got an  \x05 \x41 Archipelago item \x05 \x40 ! \x01 It seems  \x05 \x41 important \x05 \x40 ! " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9098 :  " \x08 You got an  \x05 \x43 Archipelago item \x05 \x40 ! \x01 Doesn ' t seem like it ' s needed. " , 
							 
						 
					
						
							
								
									
										
											 
										
											
												Ocarina of Time (#64)
* first commit (not including OoT data files yet)
* added some basic options
* rule parser works now at least
* make sure to commit everything this time
* temporary change to BaseClasses for oot
* overworld location graph builds mostly correctly
* adding oot data files
* commenting out world options until later since they only existed to make the RuleParser work
* conversion functions between AP ids and OOT ids
* world graph outputs
* set scrub prices
* itempool generates, entrances connected, way too many options added
* fixed set_rules and set_shop_rules
* temp baseclasses changes
* Reaches the fill step now, old event-based system retained in case the new way breaks
* Song placements and misc fixes everywhere
* temporary changes to make oot work
* changed root exits for AP fill framework
* prevent infinite recursion due to OoT sharing usage of the address field
* age reachability works hopefully, songs are broken again
* working spoiler log generation on beatable-only
* Logic tricks implemented
* need this for logic tricks
* fixed map/compass being placed on Serenade location
* kill unreachable events before filling the world
* add a bunch of utility functions to prepare for rom patching
* move OptionList into generic options
* fixed some silly bugs with OptionList
* properly seed all random behavior (so far)
* ROM generation working
* fix hints trying to get alttp dungeon hint texts
* continue fixing hints
* add oot to network data package
* change item and location IDs to 66000 and 67000 range respectively
* push removed items to precollected items
* fixed various issues with cross-contamination with multiple world generation
* reenable glitched logic (hopefully)
* glitched world files age-check fix
* cleaned up some get_locations calls
* added token shuffle and scrub shuffle, modified some options slightly to make the parsing work
* reenable MQ dungeons
* fix forest mq exception
* made targeting style an option for now, will be cosmetic later
* reminder to move targeting to cosmetics
* some oot option maintenance
* enabled starting time of day
* fixed issue breaking shop slots in multiworld generation
* added "off" option for text shuffle and hints
* shopsanity functionality restored
* change patch file extension
* remove unnecessary utility functions + imports
* update MIT license
* change option to "patch_uncompressed_rom" instead of "compress_rom"
* compliance with new AutoWorld systems
* Kill only internal events, remove non-internal big poe event in code
* re-add the big poe event and handle it correctly
* remove extra method in Range option
* fix typo
* Starting items, starting with consumables option
* do not remove nonexistent item
* move set_shop_rules to after shop items are placed
* some cleanup
* add retries for song placement
* flagged Skull Mask and Mask of Truth as advancement items
* update OoT to use LogicMixin
* Fixed trying to assign starting items from the wrong players
* fixed song retry step
* improved option handling, comments, and starting item replacements
* DefaultOnToggle writes Yes or No to spoiler
* enable compression of output if Compress executable is present
* clean up compression
* check whether (de)compressor exists before running the process
* allow specification of rom path in host.yaml
* check if decompressed file already exists before decompressing again
* fix triforce hunt generation
* rename all the oot state functions with prefix
* OoT: mark triforce pieces as completion goal for triforce hunt
* added overworld and any-dungeon shuffle for dungeon items
* Hide most unshuffled locations and events from the list of locations in spoiler
* build oot option ranges with a generic function instead of defining each separately
* move oot output-type control to host.yaml instead of individual yamls
* implement dungeon song shuffle
* minor improvements to overworld dungeon item shuffle
* remove random ice trap names in shops, mostly to avoid maintaining a massive censor list
* always output patch file to folder, remove option to generate ROM in preparation for removal
* re-add the fix for infinite recursion due to not being light or dark world
* change AP-sendable to Ocarina of Time model, since the triforce piece has some extra code apparently
* oot: remove item_names and location_names
* oot: minor fixes
* oot: comment out ROM patching
* oot: only add CollectionState objects on creation if actually needed
* main entrance shuffle method and entrances-based rules
* fix entrances based rules
* disable master quest and big poe count options for client compatibility
* use get_player_name instead of get_player_names
* fix OptionList
* fix oot options for new option system
* new coop section in oot rom: expand player names to 16 bytes, write AP_PLAYER_NAME at end of PLAYER_NAMES
* fill AP player name in oot rom with 0 instead of 0xDF
* encode player name with ASCII for fixed-width
* revert oot player name array to 8 bytes per name
* remove Pierre location if fast scarecrow is on
* check player name length
* "free_scarecrow" not "fast_scarecrow"
* OoT locations now properly store the AP ID instead of the oot internal ID
* oot __version__ updates in lockstep with AP version
* pull in unmodified oot cosmetic files
* also grab JSONDump since it's needed apparently
* gather extra needed methods, modify imports
* delete cosmetics log, replace all instances of SettingsList with OOTWorld
* cosmetic options working, except for sound effects (due to ear-safe issues)
* SFX, Music, and Fanfare randomization reenabled
* move OoT data files into the worlds folder
* move Compress and Decompress into oot data folder
* Replace get_all_state with custom method to avoid the cache
* OoT ROM: increment item counter before setting incoming item/player values to 0, preventing desync issues
* set data_version to 0
* make Kokiri Sword shuffle off by default
* reenable "Random Choice" for various cosmetic options
* kill Ruto's Letter turnin if open fountain
also fix for shopsanity
* place Buy Goron/Zora Tunic first in shop shuffle
* make ice traps appear as other items instead of breaking generation
* managed to break ice traps on non-major-only
* only handle ice traps if they are on
* fix shopsanity for non-oot games, and write player name instead of player number
* light arrows hint uses player name instead of player number
* Reenable "skip child zelda" option
* fix entrances_based_rules
* fix ganondorf hint if starting with light arrows
* fix dungeonitem shuffle and shopsanity interaction
* remove has_all_of, has_any_of, count_of in BaseClasses, replace usage with has_all, has_any, has_group
* force local giveable item on ZL if skip_child_zelda and shuffle_song_items is any
* keep bosses and bombchu bowling chus out of data package
* revert workaround for infinite recursion and fix it properly
* fix shared shop id caches during patching process
* fix shop text box overflows, as much as possible
* add default oot host.yaml option
* add .apz5, .n64, .z64 to gitignore
* Properly document and name all (functioning) OOT options
* clean up some imports
* remove unnecessary files from oot's data
* fix typo in gitignore
* readd the Compress and Decompress utilities, since they are needed for generation
* cleanup of imports and some minor optimizations
* increase shop offset for item IDs to 0xCB
* remove shop item AP ids entirely
* prevent triforce pieces for other players from being received by yourself
* add "excluded" property to Location
* Hint system adapted and reenabled; hints still unseeded
* make hints deterministic with lists instead of sets
* do not allow hints to point to Light Arrows on non-vanilla bridge
* foreign locations hint as their full name in OoT rather than their region
* checkedLocations now stores hint names by player ID, so that the same location in different worlds can have hints associated
* consolidate versioning in Utils
* ice traps appear as major items rather than any progression item
* set prescription and claim check as defaults for adult trade item settings
* add oot options to playerSettings
* allow case-insensitive logic tricks in yaml
* fix oot shopsanity option formatting
* Write OoT override info even if local item, enabling local checks to show up immediately in the client
* implement CollectionState.can_live_dmg for oot glitched logic
* filter item names for invalid characters when patching shops
* make ice traps appear according to the settings of the world they are shuffled into, rather than the original world
* set hidden-spoiler items and locations with Shop items to events
* make GF carpenters, Gerudo Card, Malon, ZL, and Impa events if the relevant settings are enabled, preventing them from appearing in the client on game start
* Fix oot Glitched and No Logic generation
* fix indenting
* Greatly reduce displayed cosmetic options
* Change oot data version to 1
* add apz5 distribution to webhost
* print player name if an ALttP dungeon contains a good item for OoT world
* delete unneeded commented code
* remove OcarinaSongs import to satisfy lint
											 
										 
										
											2021-09-02 08:35:05 -04:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								KEYSANITY_MESSAGES  =  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x001C :  " \x13 \x74 \x08 You got the  \x05 \x41 Boss Key \x05 \x40 \x01 for the  \x05 \x41 Fire Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0006 :  " \x13 \x74 \x08 You got the  \x05 \x41 Boss Key \x05 \x40 \x01 for the  \x05 \x42 Forest Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x001D :  " \x13 \x74 \x08 You got the  \x05 \x41 Boss Key \x05 \x40 \x01 for the  \x05 \x43 Water Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x001E :  " \x13 \x74 \x08 You got the  \x05 \x41 Boss Key \x05 \x40 \x01 for the  \x05 \x46 Spirit Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x002A :  " \x13 \x74 \x08 You got the  \x05 \x41 Boss Key \x05 \x40 \x01 for the  \x05 \x45 Shadow Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0061 :  " \x13 \x74 \x08 You got the  \x05 \x41 Boss Key \x05 \x40 \x01 for  \x05 \x41 Ganon ' s Castle \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0062 :  " \x13 \x75 \x08 You found the  \x05 \x41 Compass \x05 \x40 \x01 for the  \x05 \x42 Deku Tree \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0063 :  " \x13 \x75 \x08 You found the  \x05 \x41 Compass \x05 \x40 \x01 for  \x05 \x41 Dodongo ' s Cavern \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0064 :  " \x13 \x75 \x08 You found the  \x05 \x41 Compass \x05 \x40 \x01 for  \x05 \x43 Jabu Jabu ' s Belly \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0065 :  " \x13 \x75 \x08 You found the  \x05 \x41 Compass \x05 \x40 \x01 for the  \x05 \x42 Forest Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x007C :  " \x13 \x75 \x08 You found the  \x05 \x41 Compass \x05 \x40 \x01 for the  \x05 \x41 Fire Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x007D :  " \x13 \x75 \x08 You found the  \x05 \x41 Compass \x05 \x40 \x01 for the  \x05 \x43 Water Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x007E :  " \x13 \x75 \x08 You found the  \x05 \x41 Compass \x05 \x40 \x01 for the  \x05 \x46 Spirit Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x007F :  " \x13 \x75 \x08 You found the  \x05 \x41 Compass \x05 \x40 \x01 for the  \x05 \x45 Shadow Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0087 :  " \x13 \x75 \x08 You found the  \x05 \x41 Compass \x05 \x40 \x01 for the  \x05 \x44 Ice Cavern \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0088 :  " \x13 \x76 \x08 You found the  \x05 \x41 Dungeon Map \x05 \x40 \x01 for the  \x05 \x42 Deku Tree \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0089 :  " \x13 \x76 \x08 You found the  \x05 \x41 Dungeon Map \x05 \x40 \x01 for  \x05 \x41 Dodongo ' s Cavern \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x008A :  " \x13 \x76 \x08 You found the  \x05 \x41 Dungeon Map \x05 \x40 \x01 for  \x05 \x43 Jabu Jabu ' s Belly \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x008B :  " \x13 \x76 \x08 You found the  \x05 \x41 Dungeon Map \x05 \x40 \x01 for the  \x05 \x42 Forest Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x008C :  " \x13 \x76 \x08 You found the  \x05 \x41 Dungeon Map \x05 \x40 \x01 for the  \x05 \x41 Fire Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x008E :  " \x13 \x76 \x08 You found the  \x05 \x41 Dungeon Map \x05 \x40 \x01 for the  \x05 \x43 Water Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x008F :  " \x13 \x76 \x08 You found the  \x05 \x41 Dungeon Map \x05 \x40 \x01 for the  \x05 \x46 Spirit Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0092 :  " \x13 \x76 \x08 You found the  \x05 \x41 Dungeon Map \x05 \x40 \x01 for the  \x05 \x44 Ice Cavern \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0093 :  " \x13 \x77 \x08 You found a  \x05 \x41 Small Key \x05 \x40 \x01 for the  \x05 \x42 Forest Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0094 :  " \x13 \x77 \x08 You found a  \x05 \x41 Small Key \x05 \x40 \x01 for the  \x05 \x41 Fire Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0095 :  " \x13 \x77 \x08 You found a  \x05 \x41 Small Key \x05 \x40 \x01 for the  \x05 \x43 Water Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x009B :  " \x13 \x77 \x08 You found a  \x05 \x41 Small Key \x05 \x40 \x01 for the  \x05 \x45 Bottom of the Well \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x009F :  " \x13 \x77 \x08 You found a  \x05 \x41 Small Key \x05 \x40 \x01 for the  \x05 \x46 Gerudo Training \x01 Grounds \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00A0 :  " \x13 \x77 \x08 You found a  \x05 \x41 Small Key \x05 \x40 \x01 for the  \x05 \x46 Gerudo ' s Fortress \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00A1 :  " \x13 \x77 \x08 You found a  \x05 \x41 Small Key \x05 \x40 \x01 for  \x05 \x41 Ganon ' s Castle \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00A2 :  " \x13 \x75 \x08 You found the  \x05 \x41 Compass \x05 \x40 \x01 for the  \x05 \x45 Bottom of the Well \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00A3 :  " \x13 \x76 \x08 You found the  \x05 \x41 Dungeon Map \x05 \x40 \x01 for the  \x05 \x45 Shadow Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00A5 :  " \x13 \x76 \x08 You found the  \x05 \x41 Dungeon Map \x05 \x40 \x01 for the  \x05 \x45 Bottom of the Well \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00A6 :  " \x13 \x77 \x08 You found a  \x05 \x41 Small Key \x05 \x40 \x01 for the  \x05 \x46 Spirit Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x00A9 :  " \x13 \x77 \x08 You found a  \x05 \x41 Small Key \x05 \x40 \x01 for the  \x05 \x45 Shadow Temple \x05 \x40 ! \x09 " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								MISC_MESSAGES  =  {  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x507B :  ( bytearray ( 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            b " \x08 I tell you, I saw him! \x04 "  \
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            b " \x08 I saw the ghostly figure of Damp \x96 \x01 "  \
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            b " the gravekeeper sinking into \x01 "  \
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            b " his grave. It looked like he was \x01 "  \
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            b " holding some kind of  \x05 \x41 treasure \x05 \x40 ! \x02 " 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            ) ,  None ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x0422 :  ( " They say that once  \x05 \x41 Morpha ' s Curse \x05 \x40 \x01 is lifted, striking  \x05 \x42 this stone \x05 \x40  can \x01 shift the tides of  \x05 \x44 Lake Hylia \x05 \x40 . \x02 " ,  0x23 ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x401C :  ( " Please find my dear  \05 \x41 Princess Ruto \x05 \x40 \x01 immediately... Zora! \x12 \x68 \x7A " ,  0x23 ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    0x9100 :  ( " I am out of goods now. \x01 Sorry! \x04 The mark that will lead you to \x01 the Spirit Temple is the  \x05 \x41 flag on \x01 the left  \x05 \x40 outside the shop. \x01 Be seeing you! \x02 " ,  0x00 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# convert byte array to an integer  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  bytes_to_int ( bytes ,  signed = False ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  int . from_bytes ( bytes ,  byteorder = ' big ' ,  signed = signed ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# convert int to an array of bytes of the given width  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  int_to_bytes ( num ,  width ,  signed = False ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  int . to_bytes ( num ,  width ,  byteorder = ' big ' ,  signed = signed ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  display_code_list ( codes ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    message  =  " " 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  code  in  codes : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        message  + =  str ( code ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  message 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  parse_control_codes ( text ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  isinstance ( text ,  list ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  =  text 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    elif  isinstance ( text ,  bytearray ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  =  list ( text ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  =  list ( text . encode ( ' utf-8 ' ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # Special characters encoded to utf-8 must be re-encoded to OoT's values for them. 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # Tuple is used due to utf-8 encoding using two bytes. 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    i  =  0 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    while  i  <  len ( bytes )  -  1 : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  ( bytes [ i ] ,  bytes [ i + 1 ] )  in  UTF8_TO_OOT_SPECIAL : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            bytes [ i ]  =  UTF8_TO_OOT_SPECIAL [ ( bytes [ i ] ,  bytes [ i + 1 ] ) ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            del  bytes [ i + 1 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        i  + =  1 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    text_codes  =  [ ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    index  =  0 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    while  index  <  len ( bytes ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        next_char  =  bytes [ index ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        data  =  0 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        index  + =  1 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  next_char  in  CONTROL_CODES : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            extra_bytes  =  CONTROL_CODES [ next_char ] [ 1 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  extra_bytes  >  0 : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                data  =  bytes_to_int ( bytes [ index  :  index  +  extra_bytes ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                index  + =  extra_bytes 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        text_code  =  Text_Code ( next_char ,  data ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        text_codes . append ( text_code ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  text_code . code  ==  0x02 :   # message end code 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            break 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  text_codes 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# holds a single character or control code of a string  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								class  Text_Code ( ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  display ( self ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  self . code  in  CONTROL_CODES : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  CONTROL_CODES [ self . code ] [ 2 ] ( self . data ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        elif  self . code  in  SPECIAL_CHARACTERS : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  SPECIAL_CHARACTERS [ self . code ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        elif  self . code  > =  0x7F : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  ' ? ' 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  chr ( self . code ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  get_python_string ( self ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  self . code  in  CONTROL_CODES : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            ret  =  ' ' 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            subdata  =  self . data 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            for  _  in  range ( 0 ,  CONTROL_CODES [ self . code ] [ 1 ] ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                ret  =  ( ' \\ x %02X '  %  ( subdata  &  0xFF ) )  +  ret 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                subdata  =  subdata  >>  8 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            ret  =  ' \\ x %02X '  %  self . code  +  ret 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  ret 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        elif  self . code  in  SPECIAL_CHARACTERS : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  ' \\ x %02X '  %  self . code 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        elif  self . code  > =  0x7F : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  ' ? ' 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  chr ( self . code ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  get_string ( self ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  self . code  in  CONTROL_CODES : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            ret  =  ' ' 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            subdata  =  self . data 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            for  _  in  range ( 0 ,  CONTROL_CODES [ self . code ] [ 1 ] ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                ret  =  chr ( subdata  &  0xFF )  +  ret 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                subdata  =  subdata  >>  8 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            ret  =  chr ( self . code )  +  ret 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  ret 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  chr ( self . code ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # writes the code to the given offset, and returns the offset of the next byte 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  size ( self ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        size  =  1 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  self . code  in  CONTROL_CODES : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            size  + =  CONTROL_CODES [ self . code ] [ 1 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  size 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # writes the code to the given offset, and returns the offset of the next byte 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  write ( self ,  rom ,  offset ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        rom . write_byte ( TEXT_START  +  offset ,  self . code ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        extra_bytes  =  0 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  self . code  in  CONTROL_CODES : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            extra_bytes  =  CONTROL_CODES [ self . code ] [ 1 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            bytes_to_write  =  int_to_bytes ( self . data ,  extra_bytes ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            rom . write_bytes ( TEXT_START  +  offset  +  1 ,  bytes_to_write ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  offset  +  1  +  extra_bytes 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  __init__ ( self ,  code ,  data ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . code  =  code 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  code  in  CONTROL_CODES : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            self . type  =  CONTROL_CODES [ code ] [ 0 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            self . type  =  ' character ' 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . data  =  data 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    __str__  =  __repr__  =  display 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# holds a single message, and all its data  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								class  Message ( ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  display ( self ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        meta_data  =  [ " # "  +  str ( self . index ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " ID: 0x "  +  " {:04x} " . format ( self . id ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " Offset: 0x "  +  " {:06x} " . format ( self . offset ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " Length: 0x "  +  " {:04x} " . format ( self . unpadded_length )  +  " /0x "  +  " {:04x} " . format ( self . length ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " Box Type:  "  +  str ( self . box_type ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " Postion:  "  +  str ( self . position ) ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  ' ,  ' . join ( meta_data )  +  ' \n '  +  self . text 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  get_python_string ( self ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        ret  =  ' ' 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for  code  in  self . text_codes : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            ret  =  ret  +  code . get_python_string ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  ret 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # check if this is an unused message that just contains it's own id as text 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  is_id_message ( self ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  self . unpadded_length  ==  5 : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            for  i  in  range ( 4 ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                code  =  self . text_codes [ i ] . code 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if  not  ( code  in  range ( ord ( ' 0 ' ) , ord ( ' 9 ' ) + 1 )  or  code  in  range ( ord ( ' A ' ) , ord ( ' F ' ) + 1 )  or  code  in  range ( ord ( ' a ' ) , ord ( ' f ' ) + 1 )  ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    return  False 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return  True 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  False 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  parse_text ( self ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . text_codes  =  parse_control_codes ( self . raw_text ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        index  =  0 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for  text_code  in  self . text_codes : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            index  + =  text_code . size ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  text_code . code  ==  0x02 :  # message end code 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                break 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  text_code . code  ==  0x07 :  # goto 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                self . has_goto  =  True 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                self . ending  =  text_code 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  text_code . code  ==  0x0A :  # keep-open 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                self . has_keep_open  =  True 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                self . ending  =  text_code 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  text_code . code  ==  0x0B :  # event 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                self . has_event  =  True 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                self . ending  =  text_code 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  text_code . code  ==  0x0E :  # fade out 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                self . has_fade  =  True 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                self . ending  =  text_code 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  text_code . code  ==  0x10 :  # ocarina 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                self . has_ocarina  =  True 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                self . ending  =  text_code 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  text_code . code  ==  0x1B :  # two choice 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                self . has_two_choice  =  True 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  text_code . code  ==  0x1C :  # three choice 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                self . has_three_choice  =  True 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . text  =  display_code_list ( self . text_codes ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . unpadded_length  =  index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  is_basic ( self ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  not  ( self . has_goto  or  self . has_keep_open  or  self . has_event  or  self . has_fade  or  self . has_ocarina  or  self . has_two_choice  or  self . has_three_choice ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # computes the size of a message, including padding 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  size ( self ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        size  =  0 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for  code  in  self . text_codes : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            size  + =  code . size ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        size  =  ( size  +  3 )  &  - 4  # align to nearest 4 bytes 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  size 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # applies whatever transformations we want to the dialogs 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  transform ( self ,  replace_ending = False ,  ending = None ,  always_allow_skip = True ,  speed_up_text = True ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        ending_codes  =  [ 0x02 ,  0x07 ,  0x0A ,  0x0B ,  0x0E ,  0x10 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        box_breaks  =  [ 0x04 ,  0x0C ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        slows_text  =  [ 0x08 ,  0x09 ,  0x14 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        text_codes  =  [ ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        # # speed the text 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  speed_up_text : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            text_codes . append ( Text_Code ( 0x08 ,  0 ) )  # allow instant 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        # write the message 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for  code  in  self . text_codes : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            # ignore ending codes if it's going to be replaced 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  replace_ending  and  code . code  in  ending_codes : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                pass 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            # ignore the "make unskippable flag" 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            elif  always_allow_skip  and  code . code  ==  0x1A : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                pass 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            # ignore anything that slows down text 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            elif  speed_up_text  and  code . code  in  slows_text : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                pass 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            elif  speed_up_text  and  code . code  in  box_breaks : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                # some special cases for text that needs to be on a timer 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if  ( self . id  ==  0x605A  or   # twinrova transformation 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    self . id  ==  0x706C  or   # raru ending text 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    self . id  ==  0x70DD  or   # ganondorf ending text 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    self . id  ==  0x7070 ) :    # zelda ending text 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    text_codes . append ( code ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    text_codes . append ( Text_Code ( 0x08 ,  0 ) )  # allow instant 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                else : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    text_codes . append ( Text_Code ( 0x04 ,  0 ) )  # un-delayed break 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    text_codes . append ( Text_Code ( 0x08 ,  0 ) )  # allow instant 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            else : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                text_codes . append ( code ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  replace_ending : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  ending : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if  speed_up_text  and  ending . code  ==  0x10 :  # ocarina 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    text_codes . append ( Text_Code ( 0x09 ,  0 ) )  # disallow instant text 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                text_codes . append ( ending )  # write special ending 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            text_codes . append ( Text_Code ( 0x02 ,  0 ) )  # write end code 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . text_codes  =  text_codes 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # writes a Message back into the rom, using the given index and offset to update the table 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # returns the offset of the next message 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  write ( self ,  rom ,  index ,  offset ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        # construct the table entry 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        id_bytes  =  int_to_bytes ( self . id ,  2 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        offset_bytes  =  int_to_bytes ( offset ,  3 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        entry  =  id_bytes  +  bytes ( [ self . opts ,  0x00 ,  0x07 ] )  +  offset_bytes 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        # write it back 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        entry_offset  =  EXTENDED_TABLE_START  +  8  *  index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        rom . write_bytes ( entry_offset ,  entry ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for  code  in  self . text_codes : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            offset  =  code . write ( rom ,  offset ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        while  offset  %  4  >  0 : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            offset  =  Text_Code ( 0x00 ,  0 ) . write ( rom ,  offset )  # pad to 4 byte align 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  offset 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  __init__ ( self ,  raw_text ,  index ,  id ,  opts ,  offset ,  length ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . raw_text  =  raw_text 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . index  =  index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . id  =  id 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . opts  =  opts   # Textbox type and y position 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . box_type  =  ( self . opts  &  0xF0 )  >>  4 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . position  =  ( self . opts  &  0x0F ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . offset  =  offset 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . length  =  length 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . has_goto  =  False 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . has_keep_open  =  False 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . has_event  =  False 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . has_fade  =  False 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . has_ocarina  =  False 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . has_two_choice  =  False 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . has_three_choice  =  False 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . ending  =  None 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . parse_text ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # read a single message from rom 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @classmethod 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  from_rom ( cls ,  rom ,  index ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        entry_offset  =  ENG_TABLE_START  +  8  *  index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        entry  =  rom . read_bytes ( entry_offset ,  8 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        next  =  rom . read_bytes ( entry_offset  +  8 ,  8 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        id  =  bytes_to_int ( entry [ 0 : 2 ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        opts  =  entry [ 2 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        offset  =  bytes_to_int ( entry [ 5 : 8 ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        length  =  bytes_to_int ( next [ 5 : 8 ] )  -  offset 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        raw_text  =  rom . read_bytes ( TEXT_START  +  offset ,  length ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  cls ( raw_text ,  index ,  id ,  opts ,  offset ,  length ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @classmethod 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  from_string ( cls ,  text ,  id = 0 ,  opts = 0x00 ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  =  list ( text . encode ( ' utf-8 ' ) )  +  [ 0x02 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        # Clean up garbage values added when encoding special characters again. 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  =  list ( filter ( lambda  a :  a  !=  194 ,  bytes ) )  # 0xC2 added before each accent char. 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        i  =  0 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        while  i  <  len ( bytes )  -  1 : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  bytes [ i ]  in  SPECIAL_CHARACTERS  and  bytes [ i ]  not  in  UTF8_TO_OOT_SPECIAL . values ( ) :  # This indicates it's one of the button chars (A button, etc). 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                # Have to delete 2 inserted garbage values. 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                del  bytes [ i - 1 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                del  bytes [ i - 2 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                i  - =  2 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            i + =  1 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  cls ( bytes ,  0 ,  id ,  opts ,  0 ,  len ( bytes )  +  1 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @classmethod 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  from_bytearray ( cls ,  bytearray ,  id = 0 ,  opts = 0x00 ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  =  list ( bytearray )  +  [ 0x02 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  cls ( bytes ,  0 ,  id ,  opts ,  0 ,  len ( bytes )  +  1 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    __str__  =  __repr__  =  display 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# wrapper for updating the text of a message, given its message id  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# if the id does not exist in the list, then it will add it  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  update_message_by_id ( messages ,  id ,  text ,  opts = None ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # get the message index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    index  =  next (  ( m . index  for  m  in  messages  if  m . id  ==  id ) ,  - 1 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # update if it was found 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  index  > =  0 : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        update_message_by_index ( messages ,  index ,  text ,  opts ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        add_message ( messages ,  text ,  id ,  opts ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# Gets the message by its ID. Returns None if the index does not exist  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  get_message_by_id ( messages ,  id ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # get the message index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    index  =  next (  ( m . index  for  m  in  messages  if  m . id  ==  id ) ,  - 1 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  index  > =  0 : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  messages [ index ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  None 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# wrapper for updating the text of a message, given its index in the list  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  update_message_by_index ( messages ,  index ,  text ,  opts = None ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  opts  is  None : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        opts  =  messages [ index ] . opts 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  isinstance ( text ,  bytearray ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        messages [ index ]  =  Message . from_bytearray ( text ,  messages [ index ] . id ,  opts ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        messages [ index ]  =  Message . from_string ( text ,  messages [ index ] . id ,  opts ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    messages [ index ] . index  =  index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# wrapper for adding a string message to a list of messages  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  add_message ( messages ,  text ,  id = 0 ,  opts = 0x00 ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  isinstance ( text ,  bytearray ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        messages . append (  Message . from_bytearray ( text ,  id ,  opts )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        messages . append (  Message . from_string ( text ,  id ,  opts )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    messages [ - 1 ] . index  =  len ( messages )  -  1 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# holds a row in the shop item table (which contains pointers to the description and purchase messages)  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								class  Shop_Item ( ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  display ( self ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        meta_data  =  [ " # "  +  str ( self . index ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " Item: 0x "  +  " {:04x} " . format ( self . get_item_id ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " Price:  "  +  str ( self . price ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " Amount:  "  +  str ( self . pieces ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " Object: 0x "  +  " {:04x} " . format ( self . object ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " Model: 0x "  +  " {:04x} " . format ( self . model ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " Description: 0x "  +  " {:04x} " . format ( self . description_message ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " Purchase: 0x "  +  " {:04x} " . format ( self . purchase_message ) , ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        func_data  =  [ 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " func1: 0x "  +  " {:08x} " . format ( self . func1 ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " func2: 0x "  +  " {:08x} " . format ( self . func2 ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " func3: 0x "  +  " {:08x} " . format ( self . func3 ) , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         " func4: 0x "  +  " {:08x} " . format ( self . func4 ) , ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  ' ,  ' . join ( meta_data )  +  ' \n '  +  ' ,  ' . join ( func_data ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # write the shop item back 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  write ( self ,  rom ,  shop_table_address ,  index ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        entry_offset  =  shop_table_address  +  0x20  *  index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  =  [ ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  + =  int_to_bytes ( self . object ,  2 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  + =  int_to_bytes ( self . model ,  2 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  + =  int_to_bytes ( self . func1 ,  4 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  + =  int_to_bytes ( self . price ,  2 ,  signed = True ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  + =  int_to_bytes ( self . pieces ,  2 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  + =  int_to_bytes ( self . description_message ,  2 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  + =  int_to_bytes ( self . purchase_message ,  2 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  + =  [ 0x00 ,  0x00 ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  + =  int_to_bytes ( self . get_item_id ,  2 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  + =  int_to_bytes ( self . func2 ,  4 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  + =  int_to_bytes ( self . func3 ,  4 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  + =  int_to_bytes ( self . func4 ,  4 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        rom . write_bytes ( entry_offset ,  bytes ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # read a single message 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  __init__ ( self ,  rom ,  shop_table_address ,  index ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        entry_offset  =  shop_table_address  +  0x20  *  index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        entry  =  rom . read_bytes ( entry_offset ,  0x20 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . index  =  index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . object  =  bytes_to_int ( entry [ 0x00 : 0x02 ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . model  =  bytes_to_int ( entry [ 0x02 : 0x04 ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . func1  =  bytes_to_int ( entry [ 0x04 : 0x08 ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . price  =  bytes_to_int ( entry [ 0x08 : 0x0A ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . pieces  =  bytes_to_int ( entry [ 0x0A : 0x0C ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . description_message  =  bytes_to_int ( entry [ 0x0C : 0x0E ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . purchase_message  =  bytes_to_int ( entry [ 0x0E : 0x10 ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        # 0x10-0x11 is always 0000 padded apparently 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . get_item_id  =  bytes_to_int ( entry [ 0x12 : 0x14 ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . func2  =  bytes_to_int ( entry [ 0x14 : 0x18 ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . func3  =  bytes_to_int ( entry [ 0x18 : 0x1C ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        self . func4  =  bytes_to_int ( entry [ 0x1C : 0x20 ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    __str__  =  __repr__  =  display 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# reads each of the shop items  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  read_shop_items ( rom ,  shop_table_address ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    shop_items  =  [ ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  index  in  range ( 0 ,  100 ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        shop_items . append (  Shop_Item ( rom ,  shop_table_address ,  index )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  shop_items 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# writes each of the shop item back into rom  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  write_shop_items ( rom ,  shop_table_address ,  shop_items ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  s  in  shop_items : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        s . write ( rom ,  shop_table_address ,  s . index ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# these are unused shop items, and contain text ids that are used elsewhere, and should not be moved  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								SHOP_ITEM_EXCEPTIONS  =  [ 0x0A ,  0x0B ,  0x11 ,  0x12 ,  0x13 ,  0x14 ,  0x29 ]  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# returns a set of all message ids used for shop items  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  get_shop_message_id_set ( shop_items ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ids  =  set ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  shop  in  shop_items : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  shop . index  not  in  SHOP_ITEM_EXCEPTIONS : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            ids . add ( shop . description_message ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            ids . add ( shop . purchase_message ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  ids 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# remove all messages that easy to tell are unused to create space in the message index table  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  remove_unused_messages ( messages ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    messages [ : ]  =  [ m  for  m  in  messages  if  not  m . is_id_message ( ) ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  index ,  m  in  enumerate ( messages ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        m . index  =  index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# takes all messages used for shop items, and moves messages from the 00xx range into the unused 80xx range  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  move_shop_item_messages ( messages ,  shop_items ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # checks if a message id is in the item message range 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  is_in_item_range ( id ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        bytes  =  int_to_bytes ( id ,  2 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  bytes [ 0 ]  ==  0x00 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # get the ids we want to move 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ids  =  set (  id  for  id  in  get_shop_message_id_set ( shop_items )  if  is_in_item_range ( id )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # update them in the message list 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  id  in  ids : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        # should be a singleton list, but in case something funky is going on, handle it as a list regardless 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        relevant_messages  =  [ message  for  message  in  messages  if  message . id  ==  id ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  len ( relevant_messages )  > =  2 : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            raise ( TypeError ( " duplicate id in move_shop_item_messages " ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for  message  in  relevant_messages : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            message . id  | =  0x8000 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # update them in the shop item list 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  shop  in  shop_items : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  is_in_item_range ( shop . description_message ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            shop . description_message  | =  0x8000 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  is_in_item_range ( shop . purchase_message ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            shop . purchase_message  | =  0x8000 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  make_player_message ( text ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    player_text  =  ' \x05 \x42 \x0F \x05 \x40 ' 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    pronoun_mapping  =  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        " You have  " :  player_text  +  "   " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        " You are  " :   player_text  +  "  is  " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        " You ' ve  " :    player_text  +  "   " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        " Your  " :      player_text  +  " ' s  " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        " You  " :       player_text  +  "   " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        " you have  " :  player_text  +  "   " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        " you are  " :   player_text  +  "  is  " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        " you ' ve  " :    player_text  +  "   " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        " your  " :      player_text  +  " ' s  " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        " you  " :       player_text  +  "   " , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    verb_mapping  =  { 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        ' obtained  ' :  ' got  ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        ' received  ' :  ' got  ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        ' learned  ' :   ' got  ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        ' borrowed  ' :  ' got  ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        ' found  ' :     ' got  ' , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    new_text  =  text 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # Replace the first instance of a 'You' with the player name 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    lower_text  =  text . lower ( ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    you_index  =  lower_text . find ( ' you ' ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  you_index  !=  - 1 : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for  find_text ,  replace_text  in  pronoun_mapping . items ( ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            # if the index do not match, then it is not the first 'You' 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if  text . find ( find_text )  ==  you_index : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                new_text  =  new_text . replace ( find_text ,  replace_text ,  1 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                break 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # because names are longer, we shorten the verbs to they fit in the textboxes better 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  find_text ,  replace_text  in  verb_mapping . items ( ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        new_text  =  new_text . replace ( find_text ,  replace_text ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    wrapped_text  =  line_wrap ( new_text ,  False ,  False ,  False ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  wrapped_text  !=  new_text : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        new_text  =  line_wrap ( new_text ,  True ,  True ,  False ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  new_text 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# reduce item message sizes and add new item messages  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# make sure to call this AFTER move_shop_item_messages()  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  update_item_messages ( messages ,  world ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    new_item_messages  =  { * * ITEM_MESSAGES ,  * * KEYSANITY_MESSAGES } 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  id ,  text  in  new_item_messages . items ( ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  len ( world . world . worlds )  >  1 : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            update_message_by_id ( messages ,  id ,  make_player_message ( text ) ,  0x23 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            update_message_by_id ( messages ,  id ,  text ,  0x23 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  id ,  ( text ,  opt )  in  MISC_MESSAGES . items ( ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        update_message_by_id ( messages ,  id ,  text ,  opt ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# run all keysanity related patching to add messages for dungeon specific items  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  add_item_messages ( messages ,  shop_items ,  world ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    move_shop_item_messages ( messages ,  shop_items ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    update_item_messages ( messages ,  world ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# reads each of the game's messages into a list of Message objects  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  read_messages ( rom ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    table_offset  =  ENG_TABLE_START 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    index  =  0 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    messages  =  [ ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    while  True : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        entry  =  rom . read_bytes ( table_offset ,  8 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        id  =  bytes_to_int ( entry [ 0 : 2 ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  id  ==  0xFFFD : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            table_offset  + =  8 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            continue  # this is only here to give an ending offset 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  id  ==  0xFFFF : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            break  # this marks the end of the table 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        messages . append (  Message . from_rom ( rom ,  index )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        index  + =  1 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        table_offset  + =  8 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  messages 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# write the messages back  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  repack_messages ( rom ,  messages ,  permutation = None ,  always_allow_skip = True ,  speed_up_text = True ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    rom . update_dmadata_record ( TEXT_START ,  TEXT_START ,  TEXT_START  +  ENG_TEXT_SIZE_LIMIT ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  permutation  is  None : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        permutation  =  range ( len ( messages ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # repack messages 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    offset  =  0 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    text_size_limit  =  ENG_TEXT_SIZE_LIMIT 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  old_index ,  new_index  in  enumerate ( permutation ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        old_message  =  messages [ old_index ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        new_message  =  messages [ new_index ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        remember_id  =  new_message . id 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        new_message . id  =  old_message . id 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        # modify message, making it represent how we want it to be written 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        new_message . transform ( True ,  old_message . ending ,  always_allow_skip ,  speed_up_text ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        # actually write the message 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        offset  =  new_message . write ( rom ,  old_index ,  offset ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        new_message . id  =  remember_id 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # raise an exception if too much is written 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # we raise it at the end so that we know how much overflow there is 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  offset  >  text_size_limit : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        raise ( TypeError ( " Message Text table is too large: 0x "  +  " {:x} " . format ( offset )  +  "  written / 0x "  +  " {:x} " . format ( ENG_TEXT_SIZE_LIMIT )  +  "  allowed. " ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # end the table 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    table_index  =  len ( messages ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    entry  =  bytes ( [ 0xFF ,  0xFD ,  0x00 ,  0x00 ,  0x07 ] )  +  int_to_bytes ( offset ,  3 ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    entry_offset  =  EXTENDED_TABLE_START  +  8  *  table_index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    rom . write_bytes ( entry_offset ,  entry ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    table_index  + =  1 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    entry_offset  =  EXTENDED_TABLE_START  +  8  *  table_index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  8  *  ( table_index  +  1 )  >  EXTENDED_TABLE_SIZE : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        raise ( TypeError ( " Message ID table is too large: 0x "  +  " {:x} " . format ( 8  *  ( table_index  +  1 ) )  +  "  written / 0x "  +  " {:x} " . format ( EXTENDED_TABLE_SIZE )  +  "  allowed. " ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    rom . write_bytes ( entry_offset ,  [ 0xFF ,  0xFF ,  0x00 ,  0x00 ,  0x00 ,  0x00 ,  0x00 ,  0x00 ] ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								# shuffles the messages in the game, making sure to keep various message types in their own group  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  shuffle_messages ( messages ,  except_hints = True ,  always_allow_skip = True ) :  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    permutation  =  [ i  for  i ,  _  in  enumerate ( messages ) ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  is_exempt ( m ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        hint_ids  =  ( 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            GOSSIP_STONE_MESSAGES  +  TEMPLE_HINTS_MESSAGES  +  LIGHT_ARROW_HINT  + 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            list ( KEYSANITY_MESSAGES . keys ( ) )  +  shuffle_messages . shop_item_messages  + 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            shuffle_messages . scrubs_message_ids  + 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            [ 0x5036 ,  0x70F5 ]  # Chicken count and poe count respectively 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        shuffle_exempt  =  [ 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            0x208D ,          # "One more lap!" for Cow in House race. 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        is_hint  =  ( except_hints  and  m . id  in  hint_ids ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        is_error_message  =  ( m . id  ==  ERROR_MESSAGE ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        is_shuffle_exempt  =  ( m . id  in  shuffle_exempt ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  ( is_hint  or  is_error_message  or  m . is_id_message ( )  or  is_shuffle_exempt ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    have_goto          =  list (  filter ( lambda  m :  not  is_exempt ( m )  and  m . has_goto ,          messages )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    have_keep_open     =  list (  filter ( lambda  m :  not  is_exempt ( m )  and  m . has_keep_open ,     messages )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    have_event         =  list (  filter ( lambda  m :  not  is_exempt ( m )  and  m . has_event ,         messages )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    have_fade          =  list (  filter ( lambda  m :  not  is_exempt ( m )  and  m . has_fade ,          messages )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    have_ocarina       =  list (  filter ( lambda  m :  not  is_exempt ( m )  and  m . has_ocarina ,       messages )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    have_two_choice    =  list (  filter ( lambda  m :  not  is_exempt ( m )  and  m . has_two_choice ,    messages )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    have_three_choice  =  list (  filter ( lambda  m :  not  is_exempt ( m )  and  m . has_three_choice ,  messages )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    basic_messages     =  list (  filter ( lambda  m :  not  is_exempt ( m )  and  m . is_basic ( ) ,        messages )  ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    def  shuffle_group ( group ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        group_permutation  =  [ i  for  i ,  _  in  enumerate ( group ) ] 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        random . shuffle ( group_permutation ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for  index_from ,  index_to  in  enumerate ( group_permutation ) : 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            permutation [ group [ index_to ] . index ]  =  group [ index_from ] . index 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # need to use 'list' to force 'map' to actually run through 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    list (  map (  shuffle_group ,  [ 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        have_goto  +  have_keep_open  +  have_event  +  have_fade  +  basic_messages , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        have_ocarina , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        have_two_choice , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        have_three_choice , 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ] ) ) 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  permutation