| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | # object to handle the smbools and optimize them | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-08 16:52:34 -04:00
										 |  |  | from ..logic.cache import Cache | 
					
						
							|  |  |  | from ..logic.smbool import SMBool, smboolFalse | 
					
						
							|  |  |  | from ..logic.helpers import Bosses | 
					
						
							|  |  |  | from ..logic.logic import Logic | 
					
						
							|  |  |  | from ..utils.doorsmanager import DoorsManager | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  | from ..utils.objectives import Objectives | 
					
						
							| 
									
										
										
										
											2023-04-08 16:52:34 -04:00
										 |  |  | from ..utils.parameters import Knows, isKnows | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | import logging | 
					
						
							| 
									
										
										
										
											2025-07-30 03:10:36 +01:00
										 |  |  | from copy import deepcopy | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | import sys | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class SMBoolManager(object): | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |     items = ['ETank', 'Missile', 'Super', 'PowerBomb', 'Bomb', 'Charge', 'Ice', 'HiJump', 'SpeedBooster', 'Wave', 'Spazer', 'SpringBall', 'Varia', 'Plasma', 'Grapple', 'Morph', 'Reserve', 'Gravity', 'XRayScope', 'SpaceJump', 'ScrewAttack', 'Nothing', 'NoEnergy', 'MotherBrain', 'Hyper', 'Gunship'] + Bosses.Golden4() + Bosses.miniBosses() | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  |     countItems = ['Missile', 'Super', 'PowerBomb', 'ETank', 'Reserve'] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |     percentItems = ['Bomb', 'Charge', 'Ice', 'HiJump', 'SpeedBooster', 'Wave', 'Spazer', 'SpringBall', 'Varia', 'Plasma', 'Grapple', 'Morph', 'Gravity', 'XRayScope', 'SpaceJump', 'ScrewAttack'] | 
					
						
							| 
									
										
											  
											
												SM: 0.4.1 Fixes and Additional Objective Options (#1859)
* first working (most of the time) progression generation for SM using VariaRandomizer's rules, items, locations and accessPoint (as regions)
* first working single-world randomized SM rom patches
* - SM now displays message when getting an item outside for someone else (fills ROM item table)
This is dependant on modifications done to sm_randomizer_rom project
* First working MultiWorld SM
* some missing things:
- player name inject in ROM and get in client
- end game get from ROM in client
- send self item to server
- add player names table in ROM
* replaced CollectionState inheritance from SMBoolManager with a composition of an array of it (required to generation more than one SM world, which is still fails but is better)
* - reenabled balancing
* post rebase fixes
* updated SmClient.py
* + added VariaRandomizer LICENSE
* + added sm_randomizer_rom project (which builds sm.ips)
* Moved VariaRandomizer and sm_randomizer_rom projects inside worlds/sm and done some cleaning
* properly revert change made to CollectionState and more cleaning
* Fixed multiworld support patch not working with VariaRandomizer's
* missing file commit
* Fixed syntax error in unused code to satisfy Linter
* Revert "Fixed multiworld support patch not working with VariaRandomizer's"
This reverts commit fb3ca18528bb331995e3d3051648c8f84d04c08b.
* many fixes and improovement
- fixed seeded generation
- fixed broken logic when more than one SM world
- added missing rules for inter-area transitions
- added basic patch presence for logic
- added DoorManager init call to reflect present patches for logic
- moved CollectionState addition out of BaseClasses into SM world
- added condition to apply progitempool presorting only if SM world is present
- set Bosses item id to None to prevent them going into multidata
- now use get_game_players
* first working (most of the time) progression generation for SM using VariaRandomizer's rules, items, locations and accessPoint (as regions)
* first working single-world randomized SM rom patches
* - SM now displays message when getting an item outside for someone else (fills ROM item table)
This is dependant on modifications done to sm_randomizer_rom project
* First working MultiWorld SM
* some missing things:
- player name inject in ROM and get in client
- end game get from ROM in client
- send self item to server
- add player names table in ROM
* replaced CollectionState inheritance from SMBoolManager with a composition of an array of it (required to generation more than one SM world, which is still fails but is better)
* - reenabled balancing
* post rebase fixes
* updated SmClient.py
* + added VariaRandomizer LICENSE
* + added sm_randomizer_rom project (which builds sm.ips)
* Moved VariaRandomizer and sm_randomizer_rom projects inside worlds/sm and done some cleaning
* properly revert change made to CollectionState and more cleaning
* Fixed multiworld support patch not working with VariaRandomizer's
* missing file commit
* Fixed syntax error in unused code to satisfy Linter
* Revert "Fixed multiworld support patch not working with VariaRandomizer's"
This reverts commit fb3ca18528bb331995e3d3051648c8f84d04c08b.
* many fixes and improovement
- fixed seeded generation
- fixed broken logic when more than one SM world
- added missing rules for inter-area transitions
- added basic patch presence for logic
- added DoorManager init call to reflect present patches for logic
- moved CollectionState addition out of BaseClasses into SM world
- added condition to apply progitempool presorting only if SM world is present
- set Bosses item id to None to prevent them going into multidata
- now use get_game_players
* Fixed multiworld support patch not working with VariaRandomizer's
Added stage_fill_hook to set morph first in progitempool
Added back VariaRandomizer's standard patches
* + added missing files from variaRandomizer project
* + added missing variaRandomizer files (custom sprites)
+ started integrating VariaRandomizer options (WIP)
* Some fixes for player and server name display
- fixed player name of 16 characters reading too far in SM client
- fixed 12 bytes SM player name limit (now 16)
- fixed server name not being displayed in SM when using server cheat ( now displays RECEIVED FROM ARCHIPELAGO)
- request: temporarly changed default seed names displayed in SM main menu to OWTCH
* Fixed Goal completion not triggering in smClient
* integrated VariaRandomizer's options into AP (WIP)
- startAP is working
- door rando is working
- skillset is working
* - fixed itemsounds.ips crash by always including nofanfare.ips into multiworld.ips (itemsounds is now always applied and "itemsounds" preset must always be "off")
* skillset are now instanced per player instead of being a singleton class
* RomPatches are now instanced per player instead of being a singleton class
* DoorManager is now instanced per player instead of being a singleton class
* - fixed the last bugs that prevented generation of >1 SM world
* fixed crash when no skillset preset is specified in randoPreset (default to "casual")
* maxDifficulty support and itemsounds removal
- added support for maxDifficulty
- removed itemsounds patch as its always applied from multiworld patch for now
* Fixed bad merge
* Post merge adaptation
* fixed player name length fix that got lost with the merge
* fixed generation with other game type than SM
* added default randoPreset json for SM in playerSettings.yaml
* fixed broken SM client following merge
* beautified json skillset presets
* Fixed ArchipelagoSmClient not building
* Fixed conflict between mutliworld patch and beam_doors_plms patch
- doorsColorsRando now working
* SM generation now outputs APBP
- Fixed paths for patches and presets when frozen
* added missing file and fixed multithreading issue
* temporarily set data_version = 0
* more work
- added support for AP starting items
- fixed client crash with gamemode being None
- patch.py "compatible_version" is now 3
* commited missing asm files
fixed start item reserve breaking game (was using bad write offset when patching)
* Nothing item are now handled game-side. the game will now skip displaying a message box for received Nothing item (but the client will still receive it).
fixed crash in SMClient when loosing connection to SNI
* fixed No Energy Item missing its ID
fixed Plando
* merge post fixes
* fixed start item Grapple, XRay and Reserve HUD, as well as graphic beams (except ice palette color)
* fixed freeze in blue brinstar caused by Varia's custom PLM not being filled with proper Multiworld PLM address (altLocsAddresses)
* fixed start item x-ray HUD display
* Fixed start items being sent by the server (is all handled in ROM)
Start items are now not removed from itempool anymore
Nothing Item is now local_items so no player will ever pickup Nothing. Doing so reduces contribution of this world to the Multiworld the more Nothing there is though.
Fixed crash (and possibly passing but broken) at generation where the static list of IPSPatches used by all SM worlds was being modified
* fixed settings that could be applied to any SM players
* fixed auth to server only using player name (now does as ALTTP to authenticate)
* - fixed End Credits broken text
* added non SM item name display
* added all supported SM options in playerSettings.yaml
* fixed locations needing a list of parent regions (now generate a region for each location with one-way exits to each (previously) parent region
did some cleaning (mainly reverts on unnecessary core classes
* minor setting fixes and tweaks
- merged Area and lightArea settings
- made missileQty, superQty and powerBombQty use value from 10 to 90 and divide value by float(10) when generating
- fixed inverted layoutPatch setting
* added option start_inventory_removes_from_pool
fixed option names formatting
fixed lint errors
small code and repo cleanup
* Hopefully fixed ROR2 that could not send any items
* - fixed missing required change to ROR2
* fixed 0 hp when respawning without having ever saved (start items were not updating the save checksum)
* fixed typo with doors_colors_rando
* fixed checksum
* added custom sprites for off-world items (progression or not)
the original AP sprite was made with PierRoulette's SM Item Sprite Utility by ijwu
* - added missing change following upstream merge
- changed patch filename extension from apbp to apm3 so patch can be used with the new client
* added morph placement options: early means local and sphere 1
* fixed failing unit tests
* - fixed broken custom_preset options
* - big cleanup to remove unnecessary or unsupported features
* - more cleanup
* - moved sm_randomizer_rom and all always applied patches into an external project that outputs basepatch.ips
- small cleanup
* - added comment to refer to project for generating basepatch.ips (https://github.com/lordlou/SMBasepatch)
* fixed g4_skip patch that can be not applied if hud is enabled
* - fixed off world sprite that can have broken graphics (restricted to use only first 2 palette)
* - updated basepatch to reflect g4_skip removal
- moved more asm files to SMBasepatch project
* - tourian grey doors at baby metroid are now always flashing (allowing to go back if needed)
* fixed wrong path if using built as exe
* - cleaned exposed maxDifficulty options
- removed always enabled Knows
* Merged LttPClient and SMClient into SNIClient
* added varia_custom Preset Option that fetch a preset (read from a new varia_custom_preset Option) from varia's web service
* small doc precision
* - added death_link support
- fixed broken Goal Completion
- post merge fix
* - removed now useless presets
* - fixed bad internal mapping with maxDiff
- increases maxDiff if only Bosses is preventing beating the game
* - added support for lowercase custom preset sections (knows, settings and controller)
- fixed controller settings not applying to ROM
* - fixed death loop when dying with Door rando, bomb or speed booster as starting items
- varia's backup save should now be usable (automatically enabled when doing door rando)
* -added docstring for generated yaml
* fixed bad merge
* fixed broken infinity max difficulty
* commented debug prints
* adjusted credits to mark progression speed and difficulty as Non Available
* added support for more than 255 players (will print Archipelago for higher player number)
* fixed missing cleanup
* added support for 65535 different player names in ROM
* fixed generations failing when only bosses are unreachable
* - replaced setting maxDiff to infinity with a bool only affecting boss logics if only bosses are left to finish
* fixed failling generations when using 'fun' settings
Accessibility checks are forced to 'items' if restricted locations are used by VARIA following usage of 'fun' settings
* fixed debug logger
* removed unsupported "suits_restriction" option
* fixed generations failing when only bosses are unreachable (using a less intrusive approach for AP)
* - fixed deathlink emptying reserves
- added death_link_survive option that lets player survive when receiving a deathlink if the have non-empty reserves
* - merged death_link and death_link_survive options
* fixed death_link
* added a fallback default starting location instead of failing generation if an invalid one was chosen
* added Nothing and NoEnergy as hint blacklist
added missing NoEnergy as local items and removed it from progression
* fixed broken Item links
* fixed failing generation that could happen with Disabled Tourian
fixed shared Location list that could be modified for each world
* added missing force disable of EscapeRando if an escape solution cant be found
* fixed broken animal surprise patches
* prevent receiving items when in the first room of Ceres (message box in mode7 is broken)
* fixed generating with "activate chozo robots" Objective
* added soft reset that saves to initial starting location
reverted code change applied to fix softlocks from comeback checks
reverted forcing all beam local when using door rando
* replaced "save and reset" with "save and fast reload" (using same Start+Select+L+R)
* added documentation about Save and Reload
removed forgotten docstring about forcing beams as local items when using door rando
* fixed frequent failing generation on WebHost (KeyError: 'Kraid')
* added "objectiveRandom", "nbObjective", objectiveList and adapted Objective selection options to better reflect VARIA's.
fixed "collect 100% items" not being excluded when objectiveRandom is used
added Exception when VARIA initial layout fails
* fixed broken non-AP items
fixed determinism caused by the use of a set
* fixed generation failing on Webhost with string as a OptionSet (replaced default with a list of string)
cleaned doc and naming of Objective related Options
											
										 
											2023-06-29 08:51:09 -04:00
										 |  |  |     def __init__(self, player=0, maxDiff=sys.maxsize, onlyBossLeft = False): | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  |         self._items = { } | 
					
						
							|  |  |  |         self._counts = { } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.player = player | 
					
						
							|  |  |  |         self.maxDiff = maxDiff | 
					
						
							| 
									
										
										
										
											2021-12-02 00:11:42 -05:00
										 |  |  |         self.onlyBossLeft = onlyBossLeft | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # cache related | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |         #self.cacheKey = 0 | 
					
						
							|  |  |  |         #self.computeItemsPositions() | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  |         Cache.reset() | 
					
						
							|  |  |  |         Logic.factory('vanilla') | 
					
						
							|  |  |  |         self.helpers = Logic.HelpersGraph(self) | 
					
						
							|  |  |  |         self.doorsManager = DoorsManager() | 
					
						
							| 
									
										
											  
											
												SM: 0.4.1 Fixes and Additional Objective Options (#1859)
* first working (most of the time) progression generation for SM using VariaRandomizer's rules, items, locations and accessPoint (as regions)
* first working single-world randomized SM rom patches
* - SM now displays message when getting an item outside for someone else (fills ROM item table)
This is dependant on modifications done to sm_randomizer_rom project
* First working MultiWorld SM
* some missing things:
- player name inject in ROM and get in client
- end game get from ROM in client
- send self item to server
- add player names table in ROM
* replaced CollectionState inheritance from SMBoolManager with a composition of an array of it (required to generation more than one SM world, which is still fails but is better)
* - reenabled balancing
* post rebase fixes
* updated SmClient.py
* + added VariaRandomizer LICENSE
* + added sm_randomizer_rom project (which builds sm.ips)
* Moved VariaRandomizer and sm_randomizer_rom projects inside worlds/sm and done some cleaning
* properly revert change made to CollectionState and more cleaning
* Fixed multiworld support patch not working with VariaRandomizer's
* missing file commit
* Fixed syntax error in unused code to satisfy Linter
* Revert "Fixed multiworld support patch not working with VariaRandomizer's"
This reverts commit fb3ca18528bb331995e3d3051648c8f84d04c08b.
* many fixes and improovement
- fixed seeded generation
- fixed broken logic when more than one SM world
- added missing rules for inter-area transitions
- added basic patch presence for logic
- added DoorManager init call to reflect present patches for logic
- moved CollectionState addition out of BaseClasses into SM world
- added condition to apply progitempool presorting only if SM world is present
- set Bosses item id to None to prevent them going into multidata
- now use get_game_players
* first working (most of the time) progression generation for SM using VariaRandomizer's rules, items, locations and accessPoint (as regions)
* first working single-world randomized SM rom patches
* - SM now displays message when getting an item outside for someone else (fills ROM item table)
This is dependant on modifications done to sm_randomizer_rom project
* First working MultiWorld SM
* some missing things:
- player name inject in ROM and get in client
- end game get from ROM in client
- send self item to server
- add player names table in ROM
* replaced CollectionState inheritance from SMBoolManager with a composition of an array of it (required to generation more than one SM world, which is still fails but is better)
* - reenabled balancing
* post rebase fixes
* updated SmClient.py
* + added VariaRandomizer LICENSE
* + added sm_randomizer_rom project (which builds sm.ips)
* Moved VariaRandomizer and sm_randomizer_rom projects inside worlds/sm and done some cleaning
* properly revert change made to CollectionState and more cleaning
* Fixed multiworld support patch not working with VariaRandomizer's
* missing file commit
* Fixed syntax error in unused code to satisfy Linter
* Revert "Fixed multiworld support patch not working with VariaRandomizer's"
This reverts commit fb3ca18528bb331995e3d3051648c8f84d04c08b.
* many fixes and improovement
- fixed seeded generation
- fixed broken logic when more than one SM world
- added missing rules for inter-area transitions
- added basic patch presence for logic
- added DoorManager init call to reflect present patches for logic
- moved CollectionState addition out of BaseClasses into SM world
- added condition to apply progitempool presorting only if SM world is present
- set Bosses item id to None to prevent them going into multidata
- now use get_game_players
* Fixed multiworld support patch not working with VariaRandomizer's
Added stage_fill_hook to set morph first in progitempool
Added back VariaRandomizer's standard patches
* + added missing files from variaRandomizer project
* + added missing variaRandomizer files (custom sprites)
+ started integrating VariaRandomizer options (WIP)
* Some fixes for player and server name display
- fixed player name of 16 characters reading too far in SM client
- fixed 12 bytes SM player name limit (now 16)
- fixed server name not being displayed in SM when using server cheat ( now displays RECEIVED FROM ARCHIPELAGO)
- request: temporarly changed default seed names displayed in SM main menu to OWTCH
* Fixed Goal completion not triggering in smClient
* integrated VariaRandomizer's options into AP (WIP)
- startAP is working
- door rando is working
- skillset is working
* - fixed itemsounds.ips crash by always including nofanfare.ips into multiworld.ips (itemsounds is now always applied and "itemsounds" preset must always be "off")
* skillset are now instanced per player instead of being a singleton class
* RomPatches are now instanced per player instead of being a singleton class
* DoorManager is now instanced per player instead of being a singleton class
* - fixed the last bugs that prevented generation of >1 SM world
* fixed crash when no skillset preset is specified in randoPreset (default to "casual")
* maxDifficulty support and itemsounds removal
- added support for maxDifficulty
- removed itemsounds patch as its always applied from multiworld patch for now
* Fixed bad merge
* Post merge adaptation
* fixed player name length fix that got lost with the merge
* fixed generation with other game type than SM
* added default randoPreset json for SM in playerSettings.yaml
* fixed broken SM client following merge
* beautified json skillset presets
* Fixed ArchipelagoSmClient not building
* Fixed conflict between mutliworld patch and beam_doors_plms patch
- doorsColorsRando now working
* SM generation now outputs APBP
- Fixed paths for patches and presets when frozen
* added missing file and fixed multithreading issue
* temporarily set data_version = 0
* more work
- added support for AP starting items
- fixed client crash with gamemode being None
- patch.py "compatible_version" is now 3
* commited missing asm files
fixed start item reserve breaking game (was using bad write offset when patching)
* Nothing item are now handled game-side. the game will now skip displaying a message box for received Nothing item (but the client will still receive it).
fixed crash in SMClient when loosing connection to SNI
* fixed No Energy Item missing its ID
fixed Plando
* merge post fixes
* fixed start item Grapple, XRay and Reserve HUD, as well as graphic beams (except ice palette color)
* fixed freeze in blue brinstar caused by Varia's custom PLM not being filled with proper Multiworld PLM address (altLocsAddresses)
* fixed start item x-ray HUD display
* Fixed start items being sent by the server (is all handled in ROM)
Start items are now not removed from itempool anymore
Nothing Item is now local_items so no player will ever pickup Nothing. Doing so reduces contribution of this world to the Multiworld the more Nothing there is though.
Fixed crash (and possibly passing but broken) at generation where the static list of IPSPatches used by all SM worlds was being modified
* fixed settings that could be applied to any SM players
* fixed auth to server only using player name (now does as ALTTP to authenticate)
* - fixed End Credits broken text
* added non SM item name display
* added all supported SM options in playerSettings.yaml
* fixed locations needing a list of parent regions (now generate a region for each location with one-way exits to each (previously) parent region
did some cleaning (mainly reverts on unnecessary core classes
* minor setting fixes and tweaks
- merged Area and lightArea settings
- made missileQty, superQty and powerBombQty use value from 10 to 90 and divide value by float(10) when generating
- fixed inverted layoutPatch setting
* added option start_inventory_removes_from_pool
fixed option names formatting
fixed lint errors
small code and repo cleanup
* Hopefully fixed ROR2 that could not send any items
* - fixed missing required change to ROR2
* fixed 0 hp when respawning without having ever saved (start items were not updating the save checksum)
* fixed typo with doors_colors_rando
* fixed checksum
* added custom sprites for off-world items (progression or not)
the original AP sprite was made with PierRoulette's SM Item Sprite Utility by ijwu
* - added missing change following upstream merge
- changed patch filename extension from apbp to apm3 so patch can be used with the new client
* added morph placement options: early means local and sphere 1
* fixed failing unit tests
* - fixed broken custom_preset options
* - big cleanup to remove unnecessary or unsupported features
* - more cleanup
* - moved sm_randomizer_rom and all always applied patches into an external project that outputs basepatch.ips
- small cleanup
* - added comment to refer to project for generating basepatch.ips (https://github.com/lordlou/SMBasepatch)
* fixed g4_skip patch that can be not applied if hud is enabled
* - fixed off world sprite that can have broken graphics (restricted to use only first 2 palette)
* - updated basepatch to reflect g4_skip removal
- moved more asm files to SMBasepatch project
* - tourian grey doors at baby metroid are now always flashing (allowing to go back if needed)
* fixed wrong path if using built as exe
* - cleaned exposed maxDifficulty options
- removed always enabled Knows
* Merged LttPClient and SMClient into SNIClient
* added varia_custom Preset Option that fetch a preset (read from a new varia_custom_preset Option) from varia's web service
* small doc precision
* - added death_link support
- fixed broken Goal Completion
- post merge fix
* - removed now useless presets
* - fixed bad internal mapping with maxDiff
- increases maxDiff if only Bosses is preventing beating the game
* - added support for lowercase custom preset sections (knows, settings and controller)
- fixed controller settings not applying to ROM
* - fixed death loop when dying with Door rando, bomb or speed booster as starting items
- varia's backup save should now be usable (automatically enabled when doing door rando)
* -added docstring for generated yaml
* fixed bad merge
* fixed broken infinity max difficulty
* commented debug prints
* adjusted credits to mark progression speed and difficulty as Non Available
* added support for more than 255 players (will print Archipelago for higher player number)
* fixed missing cleanup
* added support for 65535 different player names in ROM
* fixed generations failing when only bosses are unreachable
* - replaced setting maxDiff to infinity with a bool only affecting boss logics if only bosses are left to finish
* fixed failling generations when using 'fun' settings
Accessibility checks are forced to 'items' if restricted locations are used by VARIA following usage of 'fun' settings
* fixed debug logger
* removed unsupported "suits_restriction" option
* fixed generations failing when only bosses are unreachable (using a less intrusive approach for AP)
* - fixed deathlink emptying reserves
- added death_link_survive option that lets player survive when receiving a deathlink if the have non-empty reserves
* - merged death_link and death_link_survive options
* fixed death_link
* added a fallback default starting location instead of failing generation if an invalid one was chosen
* added Nothing and NoEnergy as hint blacklist
added missing NoEnergy as local items and removed it from progression
* fixed broken Item links
* fixed failing generation that could happen with Disabled Tourian
fixed shared Location list that could be modified for each world
* added missing force disable of EscapeRando if an escape solution cant be found
* fixed broken animal surprise patches
* prevent receiving items when in the first room of Ceres (message box in mode7 is broken)
* fixed generating with "activate chozo robots" Objective
* added soft reset that saves to initial starting location
reverted code change applied to fix softlocks from comeback checks
reverted forcing all beam local when using door rando
* replaced "save and reset" with "save and fast reload" (using same Start+Select+L+R)
* added documentation about Save and Reload
removed forgotten docstring about forcing beams as local items when using door rando
* fixed frequent failing generation on WebHost (KeyError: 'Kraid')
* added "objectiveRandom", "nbObjective", objectiveList and adapted Objective selection options to better reflect VARIA's.
fixed "collect 100% items" not being excluded when objectiveRandom is used
added Exception when VARIA initial layout fails
* fixed broken non-AP items
fixed determinism caused by the use of a set
* fixed generation failing on Webhost with string as a OptionSet (replaced default with a list of string)
cleaned doc and naming of Objective related Options
											
										 
											2023-06-29 08:51:09 -04:00
										 |  |  |         self.objectives = Objectives.objDict[player] if player in Objectives.objDict.keys() else Objectives(player) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  |         self.createFacadeFunctions() | 
					
						
							|  |  |  |         self.createKnowsFunctions(player) | 
					
						
							|  |  |  |         self.resetItems() | 
					
						
							| 
									
										
										
										
											2025-07-30 03:10:36 +01:00
										 |  |  |         self.itemsPositions = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __deepcopy__(self, memodict): | 
					
						
							|  |  |  |         # Use __new__ to avoid calling __init__ like copy.deepcopy without __deepcopy__ implemented. | 
					
						
							|  |  |  |         new = object.__new__(type(self)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Copy everything over in the same order as __init__, ensuring that mutable attributes are deeply copied. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # SMBool instances contain mutable lists, so must be deep-copied. | 
					
						
							|  |  |  |         new._items = {i: deepcopy(v, memodict) for i, v in self._items.items()} | 
					
						
							|  |  |  |         # `_counts` is a dict[str, int], so the dict can be copied because its keys and values are immutable. | 
					
						
							|  |  |  |         new._counts = self._counts.copy() | 
					
						
							|  |  |  |         # `player` is an int. | 
					
						
							|  |  |  |         new.player = self.player | 
					
						
							|  |  |  |         # `maxDiff` is an int. | 
					
						
							|  |  |  |         new.maxDiff = self.maxDiff | 
					
						
							|  |  |  |         # `onlyBossLeft` is a bool. | 
					
						
							|  |  |  |         new.onlyBossLeft = self.onlyBossLeft | 
					
						
							|  |  |  |         # The HelpersGraph keeps reference to the instance, so a new HelpersGraph is required. | 
					
						
							|  |  |  |         new.helpers = Logic.HelpersGraph(new) | 
					
						
							|  |  |  |         # DoorsManager is stateless, so the same instance can be used. | 
					
						
							|  |  |  |         new.doorsManager = self.doorsManager | 
					
						
							|  |  |  |         # Objectives are cached by self.player, so will be the same instance for the copy. | 
					
						
							|  |  |  |         new.objectives = self.objectives | 
					
						
							|  |  |  |         # Copy the facade functions from new.helpers into new.__dict__. | 
					
						
							|  |  |  |         new.createFacadeFunctions() | 
					
						
							|  |  |  |         # Copying the existing 'knows' functions from `self` to `new` is faster than re-creating all the lambdas with | 
					
						
							|  |  |  |         # `new.createKnowsFunctions(player)`. | 
					
						
							|  |  |  |         for key in Knows.__dict__.keys(): | 
					
						
							|  |  |  |             if isKnows(key): | 
					
						
							|  |  |  |                 attribute_name = "knows"+key | 
					
						
							|  |  |  |                 knows_func = getattr(self, attribute_name) | 
					
						
							|  |  |  |                 setattr(new, attribute_name, knows_func) | 
					
						
							|  |  |  |         # There is no need to call `new.resetItems()` because `_items` and `_counts` have been copied over. | 
					
						
							|  |  |  |         # new.resetItems() | 
					
						
							|  |  |  |         # itemsPositions is a `dict[str, tuple[int, int]]`, so the dict can be copied because the keys and values are | 
					
						
							|  |  |  |         # immutable. | 
					
						
							|  |  |  |         new.itemsPositions = self.itemsPositions.copy() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return new | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def computeItemsPositions(self): | 
					
						
							|  |  |  |         # compute index in cache key for each items | 
					
						
							|  |  |  |         self.itemsPositions = {} | 
					
						
							|  |  |  |         maxBitsForCountItem = 7 # 128 values with 7 bits | 
					
						
							|  |  |  |         for (i, item) in enumerate(self.countItems): | 
					
						
							|  |  |  |             pos = i*maxBitsForCountItem | 
					
						
							|  |  |  |             bitMask = (2<<(maxBitsForCountItem-1))-1 | 
					
						
							|  |  |  |             bitMask = bitMask << pos | 
					
						
							|  |  |  |             self.itemsPositions[item] = (pos, bitMask) | 
					
						
							|  |  |  |         for (i, item) in enumerate(self.items, (i+1)*maxBitsForCountItem+1): | 
					
						
							|  |  |  |             if item in self.countItems: | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             self.itemsPositions[item] = (i, 1<<i) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def computeNewCacheKey(self, item, value): | 
					
						
							|  |  |  |         # generate an unique integer for each items combinations which is use as key in the cache. | 
					
						
							|  |  |  |         if item in ['Nothing', 'NoEnergy']: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         (pos, bitMask) = self.itemsPositions[item] | 
					
						
							|  |  |  | #        print("--------------------- {} {} ----------------------------".format(item, value)) | 
					
						
							|  |  |  | #        print("old:  "+format(self.cacheKey, '#067b')) | 
					
						
							|  |  |  |         self.cacheKey = (self.cacheKey & (~bitMask)) | (value<<pos) | 
					
						
							|  |  |  | #        print("new:  "+format(self.cacheKey, '#067b')) | 
					
						
							|  |  |  | #        self.printItemsInKey(self.cacheKey) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def printItemsInKey(self, key): | 
					
						
							|  |  |  |         # for debug purpose | 
					
						
							|  |  |  |         print("key:  "+format(key, '#067b')) | 
					
						
							|  |  |  |         msg = "" | 
					
						
							|  |  |  |         for (item, (pos, bitMask)) in self.itemsPositions.items(): | 
					
						
							|  |  |  |             value = (key & bitMask) >> pos | 
					
						
							|  |  |  |             if value != 0: | 
					
						
							|  |  |  |                 msg += " {}: {}".format(item, value) | 
					
						
							|  |  |  |         print("items:{}".format(msg)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def isEmpty(self): | 
					
						
							|  |  |  |         for item in self.items: | 
					
						
							|  |  |  |             if self.haveItem(item): | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  |         for item in self.countItems: | 
					
						
							|  |  |  |             if self.itemCount(item) > 0: | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def getItems(self): | 
					
						
							|  |  |  |         # get a dict of collected items and how many (to be displayed on the solver spoiler) | 
					
						
							|  |  |  |         itemsDict = {} | 
					
						
							|  |  |  |         for item in self.items: | 
					
						
							|  |  |  |             itemsDict[item] = 1 if self._items[item] == True else 0 | 
					
						
							|  |  |  |         for item in self.countItems: | 
					
						
							|  |  |  |             itemsDict[item] = self._counts[item] | 
					
						
							|  |  |  |         return itemsDict | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def withItem(self, item, func): | 
					
						
							| 
									
										
										
										
											2023-04-16 23:46:19 -04:00
										 |  |  |         addAndRemoveItem = self.isCountItem(item) or not self.haveItem(item) | 
					
						
							|  |  |  |         if addAndRemoveItem: | 
					
						
							|  |  |  |             self.addItem(item) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  |         ret = func(self) | 
					
						
							| 
									
										
										
										
											2023-04-16 23:46:19 -04:00
										 |  |  |         if addAndRemoveItem: | 
					
						
							|  |  |  |             self.removeItem(item) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  |         return ret | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def resetItems(self): | 
					
						
							|  |  |  |         self._items = { item : smboolFalse for item in self.items } | 
					
						
							|  |  |  |         self._counts = { item : 0 for item in self.countItems } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |         #self.cacheKey = 0 | 
					
						
							|  |  |  |         #Cache.update(self.cacheKey) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def addItem(self, item): | 
					
						
							|  |  |  |         # a new item is available | 
					
						
							|  |  |  |         self._items[item] = SMBool(True, items=[item]) | 
					
						
							|  |  |  |         if self.isCountItem(item): | 
					
						
							|  |  |  |             count = self._counts[item] + 1 | 
					
						
							|  |  |  |             self._counts[item] = count | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |             #self.computeNewCacheKey(item, count) | 
					
						
							|  |  |  |         #else: | 
					
						
							|  |  |  |             #self.computeNewCacheKey(item, 1) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |         #Cache.update(self.cacheKey) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def addItems(self, items): | 
					
						
							|  |  |  |         if len(items) == 0: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         for item in items: | 
					
						
							|  |  |  |             self._items[item] = SMBool(True, items=[item]) | 
					
						
							|  |  |  |             if self.isCountItem(item): | 
					
						
							|  |  |  |                 count = self._counts[item] + 1 | 
					
						
							|  |  |  |                 self._counts[item] = count | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |                 #self.computeNewCacheKey(item, count) | 
					
						
							|  |  |  |             #else: | 
					
						
							|  |  |  |                 #self.computeNewCacheKey(item, 1) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |         #Cache.update(self.cacheKey) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def removeItem(self, item): | 
					
						
							|  |  |  |         # randomizer removed an item (or the item was added to test a post available) | 
					
						
							|  |  |  |         if self.isCountItem(item): | 
					
						
							|  |  |  |             count = self._counts[item] - 1 | 
					
						
							|  |  |  |             self._counts[item] = count | 
					
						
							|  |  |  |             if count == 0: | 
					
						
							|  |  |  |                 self._items[item] = smboolFalse | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |             #self.computeNewCacheKey(item, count) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  |         else: | 
					
						
							|  |  |  |             self._items[item] = smboolFalse | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |             #self.computeNewCacheKey(item, 0) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |         #Cache.update(self.cacheKey) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def createFacadeFunctions(self): | 
					
						
							|  |  |  |         for fun in dir(self.helpers): | 
					
						
							|  |  |  |             if fun != 'smbm' and fun[0:2] != '__': | 
					
						
							|  |  |  |                 setattr(self, fun, getattr(self.helpers, fun)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def traverse(self, doorName): | 
					
						
							|  |  |  |         return self.doorsManager.traverse(self, doorName) | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |      | 
					
						
							|  |  |  |     def canPassG4(self): | 
					
						
							|  |  |  |         return self.objectives.canClearGoals(self, 'Golden Four') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def hasItemsPercent(self, percent, totalItemsCount=None): | 
					
						
							|  |  |  |         if totalItemsCount is None: | 
					
						
							|  |  |  |             totalItemsCount = self.objectives.getTotalItemsCount() | 
					
						
							|  |  |  |         currentItemsCount = self.getCollectedItemsCount() | 
					
						
							|  |  |  |         return SMBool(100*(currentItemsCount/totalItemsCount) >= percent) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def getCollectedItemsCount(self): | 
					
						
							|  |  |  |         return (len([item for item in self._items if self.haveItem(item) and item in self.percentItems]) | 
					
						
							|  |  |  |                 + sum([self.itemCount(item) for item in self._items if self.isCountItem(item)])) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def createKnowsFunctions(self, player): | 
					
						
							|  |  |  |         # for each knows we have a function knowsKnows (ex: knowsAlcatrazEscape()) which | 
					
						
							|  |  |  |         # take no parameter | 
					
						
							|  |  |  |         for knows in Knows.__dict__: | 
					
						
							|  |  |  |             if isKnows(knows): | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |                 self._createKnowsFunction(knows, player) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _setKnowsFunction(self, knows, k): | 
					
						
							|  |  |  |         setattr(self, 'knows'+knows, lambda: SMBool(k.bool, k.difficulty, | 
					
						
							|  |  |  |                                                     knows=[knows])) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _createKnowsFunction(self, knows, player): | 
					
						
							|  |  |  |         if player in Knows.knowsDict and knows in Knows.knowsDict[player].__dict__: | 
					
						
							|  |  |  |             self._setKnowsFunction(knows, Knows.knowsDict[player].__dict__[knows]) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self._setKnowsFunction(knows, Knows.__dict__[knows]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def changeKnows(self, knows, newVal): | 
					
						
							|  |  |  |         if isKnows(knows): | 
					
						
							|  |  |  |             self._setKnowsFunction(knows, newVal) | 
					
						
							|  |  |  |             #Cache.reset() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             raise ValueError("Invalid knows "+str(knows)) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |     def restoreKnows(self, knows): | 
					
						
							|  |  |  |         if isKnows(knows): | 
					
						
							|  |  |  |             self._createKnowsFunction(knows) | 
					
						
							|  |  |  |             #Cache.reset() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             raise ValueError("Invalid knows "+str(knows)) | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  |     def isCountItem(self, item): | 
					
						
							|  |  |  |         return item in self.countItems | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def itemCount(self, item): | 
					
						
							|  |  |  |         # return integer | 
					
						
							|  |  |  |         #self.state.item_count(item, self.player) | 
					
						
							|  |  |  |         return self._counts[item] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def haveItem(self, item): | 
					
						
							|  |  |  |         #return self.state.has(item, self.player) | 
					
						
							|  |  |  |         return self._items[item] | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |      | 
					
						
							|  |  |  |     def haveItems(self, items): | 
					
						
							|  |  |  |         for item in items: | 
					
						
							|  |  |  |             if not self.haveItem(item): | 
					
						
							|  |  |  |                 return smboolFalse | 
					
						
							|  |  |  |         return SMBool(True) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     wand = staticmethod(SMBool.wand) | 
					
						
							|  |  |  |     wandmax = staticmethod(SMBool.wandmax) | 
					
						
							|  |  |  |     wor = staticmethod(SMBool.wor) | 
					
						
							|  |  |  |     wnot = staticmethod(SMBool.wnot) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def itemCountOk(self, item, count, difficulty=0): | 
					
						
							|  |  |  |         if self.itemCount(item) >= count: | 
					
						
							|  |  |  |             if item in ['ETank', 'Reserve']: | 
					
						
							|  |  |  |                 item = str(count)+'-'+item | 
					
						
							|  |  |  |             return SMBool(True, difficulty, items = [item]) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return smboolFalse | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def energyReserveCountOk(self, count, difficulty=0): | 
					
						
							|  |  |  |         if self.energyReserveCount() >= count: | 
					
						
							|  |  |  |             nEtank = self.itemCount('ETank') | 
					
						
							|  |  |  |             if nEtank > count: | 
					
						
							|  |  |  |                 nEtank = int(count) | 
					
						
							|  |  |  |             items = str(nEtank)+'-ETank' | 
					
						
							|  |  |  |             nReserve = self.itemCount('Reserve') | 
					
						
							|  |  |  |             if nEtank < count: | 
					
						
							|  |  |  |                 nReserve = int(count) - nEtank | 
					
						
							|  |  |  |                 items += ' - '+str(nReserve)+'-Reserve' | 
					
						
							|  |  |  |             return SMBool(True, difficulty, items = [items]) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return smboolFalse | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class SMBoolManagerPlando(SMBoolManager): | 
					
						
							|  |  |  |     def __init__(self): | 
					
						
							|  |  |  |         super(SMBoolManagerPlando, self).__init__() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-30 03:10:36 +01:00
										 |  |  |     def __deepcopy__(self, memodict): | 
					
						
							|  |  |  |         return super().__deepcopy__(memodict) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  |     def addItem(self, item): | 
					
						
							|  |  |  |         # a new item is available | 
					
						
							|  |  |  |         already = self.haveItem(item) | 
					
						
							|  |  |  |         isCount = self.isCountItem(item) | 
					
						
							|  |  |  |         if isCount or not already: | 
					
						
							|  |  |  |             self._items[item] = SMBool(True, items=[item]) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             # handle duplicate major items (plandos) | 
					
						
							|  |  |  |             self._items['dup_'+item] = True | 
					
						
							|  |  |  |         if isCount: | 
					
						
							|  |  |  |             count = self._counts[item] + 1 | 
					
						
							|  |  |  |             self._counts[item] = count | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |             #self.computeNewCacheKey(item, count) | 
					
						
							|  |  |  |         #else: | 
					
						
							|  |  |  |             #self.computeNewCacheKey(item, 1) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |         #Cache.update(self.cacheKey) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def removeItem(self, item): | 
					
						
							|  |  |  |         # randomizer removed an item (or the item was added to test a post available) | 
					
						
							|  |  |  |         if self.isCountItem(item): | 
					
						
							|  |  |  |             count = self._counts[item] - 1 | 
					
						
							|  |  |  |             self._counts[item] = count | 
					
						
							|  |  |  |             if count == 0: | 
					
						
							|  |  |  |                 self._items[item] = smboolFalse | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |             #self.computeNewCacheKey(item, count) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  |         else: | 
					
						
							|  |  |  |             dup = 'dup_'+item | 
					
						
							|  |  |  |             if self._items.get(dup, None) is None: | 
					
						
							|  |  |  |                 self._items[item] = smboolFalse | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |                 #self.computeNewCacheKey(item, 0) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 del self._items[dup] | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |                 #self.computeNewCacheKey(item, 1) | 
					
						
							| 
									
										
										
										
											2021-11-12 08:00:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-09 18:35:46 -04:00
										 |  |  |         #Cache.update(self.cacheKey) |