| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  | from __future__ import annotations | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-21 23:27:30 +02:00
										 |  |  |  | import functools | 
					
						
							|  |  |  |  | import numbers | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  | import random | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  | from dataclasses import dataclass | 
					
						
							|  |  |  |  | from itertools import accumulate, chain, combinations | 
					
						
							|  |  |  |  | from typing import Any, cast, Dict, Iterator, List, Mapping, Optional, Set, Tuple, Type, TYPE_CHECKING, Union | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-21 23:27:30 +02:00
										 |  |  |  | from Options import AssembleOptions, Choice, DeathLink, ItemDict, OptionDict, PerGameCommonOptions, Range, \ | 
					
						
							|  |  |  |  |     SpecialRange, TextChoice, Toggle | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  | from .Enemies import enemy_name_to_sprite | 
					
						
							| 
									
										
										
										
											2023-10-21 23:27:30 +02:00
										 |  |  |  | from .Items import ItemType, l2ac_item_table | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | if TYPE_CHECKING: | 
					
						
							|  |  |  |  |     from BaseClasses import PlandoOptions | 
					
						
							|  |  |  |  |     from worlds.AutoWorld import World | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class AssembleCustomizableChoices(AssembleOptions): | 
					
						
							|  |  |  |  |     def __new__(mcs, name: str, bases: Tuple[type, ...], attrs: Dict[str, Any]) -> AssembleCustomizableChoices: | 
					
						
							|  |  |  |  |         cls: AssembleOptions = super().__new__(mcs, name, bases, attrs) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if "extra_options" in attrs: | 
					
						
							|  |  |  |  |             cls.name_lookup.update(enumerate(attrs["extra_options"], start=max(cls.name_lookup) + 1)) | 
					
						
							|  |  |  |  |         return cast(AssembleCustomizableChoices, cls) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class RandomGroupsChoice(Choice, metaclass=AssembleCustomizableChoices): | 
					
						
							|  |  |  |  |     extra_options: Optional[Set[str]] | 
					
						
							|  |  |  |  |     random_groups: Dict[str, List[str]] | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @classmethod | 
					
						
							|  |  |  |  |     def get_option_name(cls, value: int) -> str: | 
					
						
							|  |  |  |  |         if value in cls.options.values(): | 
					
						
							|  |  |  |  |             return next(k for k, v in cls.options.items() if v == value) | 
					
						
							|  |  |  |  |         else: | 
					
						
							|  |  |  |  |             return super().get_option_name(value) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @classmethod | 
					
						
							|  |  |  |  |     def from_text(cls, text: str) -> Choice: | 
					
						
							|  |  |  |  |         key: str = text.lower() | 
					
						
							|  |  |  |  |         if key == "random": | 
					
						
							|  |  |  |  |             text = random.choice([o for o in cls.options if o not in cls.random_groups]) | 
					
						
							|  |  |  |  |         elif key in cls.random_groups: | 
					
						
							|  |  |  |  |             text = random.choice(cls.random_groups[key]) | 
					
						
							|  |  |  |  |         return super().from_text(text) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  | class EnemyChoice(TextChoice): | 
					
						
							|  |  |  |  |     _valid_sprites: Dict[str, int] = {enemy_name.lower(): sprite for enemy_name, sprite in enemy_name_to_sprite.items()} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def verify(self, world: Type[World], player_name: str, plando_options: PlandoOptions) -> None: | 
					
						
							|  |  |  |  |         if isinstance(self.value, int): | 
					
						
							|  |  |  |  |             return | 
					
						
							|  |  |  |  |         if str(self.value).lower() in self._valid_sprites: | 
					
						
							|  |  |  |  |             return | 
					
						
							|  |  |  |  |         raise ValueError(f"Could not find option '{self.value}' for '{self.__class__.__name__}', known options are:\n" | 
					
						
							|  |  |  |  |                          f"{', '.join(self.options)}, {', '.join(enemy_name_to_sprite)}.") | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def sprite(self) -> Optional[int]: | 
					
						
							|  |  |  |  |         return self._valid_sprites.get(str(self.value).lower()) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  | class LevelMixin: | 
					
						
							|  |  |  |  |     xp_coefficients: List[int] = sorted([191, 65, 50, 32, 18, 14, 6, 3, 3, 2, 2, 2, 2] * 8, reverse=True) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @classmethod | 
					
						
							|  |  |  |  |     def _to_xp(cls, level: int, *, capsule: bool) -> int: | 
					
						
							|  |  |  |  |         if level == 1: | 
					
						
							|  |  |  |  |             return 0 | 
					
						
							|  |  |  |  |         if level == 99: | 
					
						
							|  |  |  |  |             return 9999999 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         increment: int = 20 << 8 | 
					
						
							|  |  |  |  |         total: int = increment | 
					
						
							|  |  |  |  |         for lv in range(2, level): | 
					
						
							|  |  |  |  |             increment += (increment * cls.xp_coefficients[lv]) >> 8 | 
					
						
							|  |  |  |  |             total += increment | 
					
						
							|  |  |  |  |             if capsule: | 
					
						
							|  |  |  |  |                 total &= 0xFFFFFF00 | 
					
						
							|  |  |  |  |         return (total >> 8) - 10 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class BlueChestChance(Range): | 
					
						
							|  |  |  |  |     """The chance of a chest being a blue chest.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     It is given in units of 1/256, i.e., a value of 25 corresponds to 25/256 ~ 9.77%. | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  |     If you increase the blue chest chance, then the red chest chance is decreased in return. | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  |     Supported values: 5 – 75 | 
					
						
							|  |  |  |  |     Default value: 25 (five times as much as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Blue chest chance" | 
					
						
							|  |  |  |  |     range_start = 5 | 
					
						
							|  |  |  |  |     range_end = 75 | 
					
						
							|  |  |  |  |     default = 25 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  |     @property | 
					
						
							|  |  |  |  |     def chest_type_thresholds(self) -> bytes: | 
					
						
							|  |  |  |  |         ratio: float = (256 - self.value) / (256 - 5) | 
					
						
							|  |  |  |  |         # unmodified chances are: consumable (mostly non-restorative) = 36/256, consumable (restorative) = 58/256, | 
					
						
							|  |  |  |  |         # blue chest = 5/256, spell = 30/256, gear = 45/256 (and the remaining part, weapon = 82/256) | 
					
						
							|  |  |  |  |         chest_type_chances: List[float] = [36 * ratio, 58 * ratio, float(self.value), 30 * ratio, 45 * ratio] | 
					
						
							|  |  |  |  |         return bytes(round(threshold) for threshold in reversed(tuple(accumulate(chest_type_chances)))) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | class BlueChestCount(Range): | 
					
						
							|  |  |  |  |     """The number of blue chest items that will be in your item pool.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     The number of blue chests in your world that count as multiworld location checks will be equal this amount plus one | 
					
						
							|  |  |  |  |     more for each party member or capsule monster if you have shuffle_party_members/shuffle_capsule_monsters enabled. | 
					
						
							|  |  |  |  |     (You will still encounter blue chests in your world after all the multiworld location checks have been exhausted, | 
					
						
							|  |  |  |  |     but these chests will then generate items for yourself only.) | 
					
						
							| 
									
										
										
										
											2023-06-29 15:06:58 +02:00
										 |  |  |  |     Supported values: 10 – 100 | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  |     Default value: 25 | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Blue chest count" | 
					
						
							|  |  |  |  |     range_start = 10 | 
					
						
							| 
									
										
										
										
											2023-06-29 15:06:58 +02:00
										 |  |  |  |     range_end = 100 | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  |     default = 25 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class Boss(RandomGroupsChoice): | 
					
						
							|  |  |  |  |     """Which boss to fight on the final floor.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: | 
					
						
							|  |  |  |  |     lizard_man, big_catfish, regal_goblin, follower_x2, camu, tarantula, pierre, daniele, gades_a, mummy_x4, troll_x3, | 
					
						
							|  |  |  |  |     gades_b, idura_a, lion_x2, idura_b, idura_c, rogue_flower, soldier_x4, gargoyle_x4, venge_ghost, white_dragon_x3, | 
					
						
							|  |  |  |  |     fire_dragon, ghost_ship, tank, gades_c, amon, erim, daos, egg_dragon, master | 
					
						
							|  |  |  |  |     random-low — select a random regular boss, from lizard_man to troll_x3 | 
					
						
							|  |  |  |  |     random-middle — select a random regular boss, from idura_a to gargoyle_x4 | 
					
						
							|  |  |  |  |     random-high — select a random regular boss, from venge_ghost to tank | 
					
						
							|  |  |  |  |     random-sinistral — select a random Sinistral boss | 
					
						
							|  |  |  |  |     Default value: master (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Boss" | 
					
						
							|  |  |  |  |     option_lizard_man = 0x01 | 
					
						
							|  |  |  |  |     option_big_catfish = 0x02 | 
					
						
							|  |  |  |  |     # 0x03 = Goblin + Skeleton; regular monsters | 
					
						
							|  |  |  |  |     # 0x04 = Goblin; regular monster | 
					
						
							|  |  |  |  |     option_regal_goblin = 0x05 | 
					
						
							|  |  |  |  |     option_follower_x2 = 0x06 | 
					
						
							|  |  |  |  |     option_camu = 0x07 | 
					
						
							|  |  |  |  |     option_tarantula = 0x08 | 
					
						
							|  |  |  |  |     option_pierre = 0x09 | 
					
						
							|  |  |  |  |     option_daniele = 0x0A | 
					
						
							|  |  |  |  |     option_gades_a = 0x0B | 
					
						
							|  |  |  |  |     option_mummy_x4 = 0x0C | 
					
						
							|  |  |  |  |     option_troll_x3 = 0x0D | 
					
						
							|  |  |  |  |     option_gades_b = 0x0E | 
					
						
							|  |  |  |  |     option_idura_a = 0x0F | 
					
						
							|  |  |  |  |     # 0x10 = Pierre; Maxim + Tia only | 
					
						
							|  |  |  |  |     # 0x11 = Daniele; Guy + Selan only | 
					
						
							|  |  |  |  |     option_lion_x2 = 0x12 | 
					
						
							|  |  |  |  |     option_idura_b = 0x13 | 
					
						
							|  |  |  |  |     option_idura_c = 0x14 | 
					
						
							|  |  |  |  |     option_rogue_flower = 0x15 | 
					
						
							|  |  |  |  |     option_soldier_x4 = 0x16 | 
					
						
							|  |  |  |  |     option_gargoyle_x4 = 0x17 | 
					
						
							|  |  |  |  |     option_venge_ghost = 0x18 | 
					
						
							|  |  |  |  |     option_white_dragon_x3 = 0x19 | 
					
						
							|  |  |  |  |     option_fire_dragon = 0x1A | 
					
						
							|  |  |  |  |     option_ghost_ship = 0x1B | 
					
						
							|  |  |  |  |     # 0x1C = Soldier x4; same as 0x16 | 
					
						
							|  |  |  |  |     # 0x1D = Soldier x4; same as 0x16 | 
					
						
							|  |  |  |  |     option_tank = 0x1E | 
					
						
							|  |  |  |  |     option_gades_c = 0x1F | 
					
						
							|  |  |  |  |     option_amon = 0x20 | 
					
						
							|  |  |  |  |     # 0x21 = Gades; same as 0x1F | 
					
						
							|  |  |  |  |     # 0x22 = Amon; same as 0x20 | 
					
						
							|  |  |  |  |     option_erim = 0x23 | 
					
						
							|  |  |  |  |     option_daos = 0x24 | 
					
						
							|  |  |  |  |     option_egg_dragon = 0x25 | 
					
						
							|  |  |  |  |     option_master = 0x26 | 
					
						
							|  |  |  |  |     default = option_master | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-30 05:21:46 +02:00
										 |  |  |  |     _sprite: Dict[int, int] = { | 
					
						
							|  |  |  |  |         option_lizard_man: 0x9E, | 
					
						
							|  |  |  |  |         option_big_catfish: 0xC5, | 
					
						
							|  |  |  |  |         option_regal_goblin: 0x9D, | 
					
						
							|  |  |  |  |         option_follower_x2: 0x76, | 
					
						
							|  |  |  |  |         option_camu: 0x75, | 
					
						
							|  |  |  |  |         option_tarantula: 0xC6, | 
					
						
							|  |  |  |  |         option_pierre: 0x77, | 
					
						
							|  |  |  |  |         option_daniele: 0x78, | 
					
						
							|  |  |  |  |         option_gades_a: 0x7A, | 
					
						
							|  |  |  |  |         option_mummy_x4: 0xA8, | 
					
						
							|  |  |  |  |         option_troll_x3: 0xA9, | 
					
						
							|  |  |  |  |         option_gades_b: 0x7A, | 
					
						
							|  |  |  |  |         option_idura_a: 0x74, | 
					
						
							|  |  |  |  |         option_lion_x2: 0xB7, | 
					
						
							|  |  |  |  |         option_idura_b: 0x74, | 
					
						
							|  |  |  |  |         option_idura_c: 0x74, | 
					
						
							|  |  |  |  |         option_rogue_flower: 0x96, | 
					
						
							|  |  |  |  |         option_soldier_x4: 0x18, | 
					
						
							|  |  |  |  |         option_gargoyle_x4: 0xC4, | 
					
						
							|  |  |  |  |         option_venge_ghost: 0xD0, | 
					
						
							|  |  |  |  |         option_white_dragon_x3: 0xC3, | 
					
						
							|  |  |  |  |         option_fire_dragon: 0xC0, | 
					
						
							|  |  |  |  |         option_ghost_ship: 0xC8, | 
					
						
							|  |  |  |  |         option_tank: 0xC7, | 
					
						
							|  |  |  |  |         option_gades_c: 0x7A, | 
					
						
							|  |  |  |  |         option_amon: 0x79, | 
					
						
							|  |  |  |  |         option_erim: 0x38, | 
					
						
							|  |  |  |  |         option_daos: 0x7B, | 
					
						
							|  |  |  |  |         option_egg_dragon: 0xC0, | 
					
						
							|  |  |  |  |         option_master: 0x94, | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  |     random_groups = { | 
					
						
							|  |  |  |  |         "random-low": ["lizard_man", "big_catfish", "regal_goblin", "follower_x2", "camu", "tarantula", "pierre", | 
					
						
							|  |  |  |  |                        "daniele", "mummy_x4", "troll_x3"], | 
					
						
							|  |  |  |  |         "random-middle": ["idura_a", "lion_x2", "idura_b", "idura_c", "rogue_flower", "soldier_x4", "gargoyle_x4"], | 
					
						
							|  |  |  |  |         "random-high": ["venge_ghost", "white_dragon_x3", "fire_dragon", "ghost_ship", "tank"], | 
					
						
							|  |  |  |  |         "random-sinistral": ["gades_c", "amon", "erim", "daos"], | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  |     extra_options = set(random_groups) | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def flag(self) -> int: | 
					
						
							|  |  |  |  |         return 0xFE if self.value == Boss.option_master else 0xFF | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-30 05:21:46 +02:00
										 |  |  |  |     @property | 
					
						
							|  |  |  |  |     def music(self) -> int: | 
					
						
							|  |  |  |  |         return 0x1B if self.value in {Boss.option_master, Boss.option_gades_a, Boss.option_gades_b, Boss.option_gades_c, | 
					
						
							|  |  |  |  |                                       Boss.option_amon, Boss.option_erim, Boss.option_daos} else 0x19 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def sprite(self) -> int: | 
					
						
							|  |  |  |  |         return Boss._sprite[self.value] | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | class CapsuleCravingsJPStyle(Toggle): | 
					
						
							|  |  |  |  |     """Make capsule monster cravings behave as in the Japanese version.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     In the US version, the data that determines which items a capsule monster can request is a mess. | 
					
						
							|  |  |  |  |     It allows only for a very limited selection of items to be requested, and the quality of the selected item is almost | 
					
						
							|  |  |  |  |     always either too low or too high (compared to the capsule monsters current quality preference). This means that, | 
					
						
							|  |  |  |  |     if fed, the requested item will either be rejected by the capsule monster or lead to an unreasonable increase of the | 
					
						
							|  |  |  |  |     quality preference, making further feeding more difficult. | 
					
						
							|  |  |  |  |     This setting provides a fix for the bug described above. | 
					
						
							|  |  |  |  |     If enabled, the capsule monster feeding behavior will be changed to behave analogous to the JP (and EU) version. | 
					
						
							|  |  |  |  |     This means that requests become more varied, while the requested item will be guaranteed to be of the same quality | 
					
						
							|  |  |  |  |     as the capsule monsters current preference. Thus, it can no longer happen that the capsule monster dislikes eating | 
					
						
							|  |  |  |  |     the very item it just requested. | 
					
						
							|  |  |  |  |     Supported values: false, true | 
					
						
							|  |  |  |  |     Default value: false (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Capsule cravings JP style" | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class CapsuleStartingForm(SpecialRange): | 
					
						
							|  |  |  |  |     """The starting form of your capsule monsters.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: 1 – 4, m | 
					
						
							|  |  |  |  |     Default value: 1 (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Capsule monster starting form" | 
					
						
							|  |  |  |  |     range_start = 1 | 
					
						
							|  |  |  |  |     range_end = 5 | 
					
						
							|  |  |  |  |     default = 1 | 
					
						
							|  |  |  |  |     special_range_cutoff = 1 | 
					
						
							|  |  |  |  |     special_range_names = { | 
					
						
							|  |  |  |  |         "default": 1, | 
					
						
							|  |  |  |  |         "m": 5, | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def unlock(self) -> int: | 
					
						
							|  |  |  |  |         if self.value == self.special_range_names["m"]: | 
					
						
							|  |  |  |  |             return 0x0B | 
					
						
							|  |  |  |  |         else: | 
					
						
							|  |  |  |  |             return self.value - 1 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class CapsuleStartingLevel(LevelMixin, SpecialRange): | 
					
						
							|  |  |  |  |     """The starting level of your capsule monsters.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Can be set to the special value party_starting_level to make it the same value as the party_starting_level option. | 
					
						
							|  |  |  |  |     Supported values: 1 – 99, party_starting_level | 
					
						
							|  |  |  |  |     Default value: 1 (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Capsule monster starting level" | 
					
						
							|  |  |  |  |     range_start = 0 | 
					
						
							|  |  |  |  |     range_end = 99 | 
					
						
							|  |  |  |  |     default = 1 | 
					
						
							|  |  |  |  |     special_range_cutoff = 1 | 
					
						
							|  |  |  |  |     special_range_names = { | 
					
						
							|  |  |  |  |         "default": 1, | 
					
						
							|  |  |  |  |         "party_starting_level": 0, | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def xp(self) -> int: | 
					
						
							|  |  |  |  |         return self._to_xp(self.value, capsule=True) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class CrowdedFloorChance(Range): | 
					
						
							|  |  |  |  |     """The chance of a floor being a crowded floor.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     It is given in units of 1/256, i.e., a value of 16 corresponds to 16/256 = 6.25%. | 
					
						
							|  |  |  |  |     A crowded floor is a floor where most of the chests are grouped in one room together with many enemies. | 
					
						
							|  |  |  |  |     Supported values: 0 – 255 | 
					
						
							|  |  |  |  |     Default value: 16 (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Crowded floor chance" | 
					
						
							|  |  |  |  |     range_start = 0 | 
					
						
							|  |  |  |  |     range_end = 255 | 
					
						
							|  |  |  |  |     default = 16 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  | class CustomItemPool(ItemDict, Mapping[str, int]): | 
					
						
							|  |  |  |  |     """Customize your multiworld item pool.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Using this option you can place any cave item in your multiworld item pool. (By default, the pool is filled with | 
					
						
							|  |  |  |  |     blue chest items.) Here you can add any valid item from the Lufia II Ancient Cave section of the datapackage | 
					
						
							|  |  |  |  |     (see https://archipelago.gg/datapackage). The value of this option has to be a mapping of item name to count, | 
					
						
							|  |  |  |  |     e.g., to add two Deadly rods and one Dekar Blade: {Deadly rod: 2, Dekar blade: 1} | 
					
						
							|  |  |  |  |     The maximum total amount of custom items you can place is limited by the chosen blue_chest_count; any remaining, | 
					
						
							|  |  |  |  |     non-customized space in the pool will be occupied by random blue chest items. | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Custom item pool" | 
					
						
							|  |  |  |  |     value: Dict[str, int] | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def count(self) -> int: | 
					
						
							|  |  |  |  |         return sum(self.values()) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __getitem__(self, key: str) -> int: | 
					
						
							|  |  |  |  |         return self.value.__getitem__(key) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __iter__(self) -> Iterator[str]: | 
					
						
							|  |  |  |  |         return self.value.__iter__() | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __len__(self) -> int: | 
					
						
							|  |  |  |  |         return self.value.__len__() | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-18 14:44:20 +01:00
										 |  |  |  | class DefaultCapsule(Choice): | 
					
						
							|  |  |  |  |     """Preselect the active capsule monster.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     (Only has an effect if shuffle_capsule_monsters is set to false.) | 
					
						
							|  |  |  |  |     Supported values: jelze, flash, gusto, zeppy, darbi, sully, blaze | 
					
						
							|  |  |  |  |     Default value: jelze | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Default capsule monster" | 
					
						
							|  |  |  |  |     option_jelze = 0x00 | 
					
						
							|  |  |  |  |     option_flash = 0x01 | 
					
						
							|  |  |  |  |     option_gusto = 0x02 | 
					
						
							|  |  |  |  |     option_zeppy = 0x03 | 
					
						
							|  |  |  |  |     option_darbi = 0x04 | 
					
						
							|  |  |  |  |     option_sully = 0x05 | 
					
						
							|  |  |  |  |     option_blaze = 0x06 | 
					
						
							|  |  |  |  |     default = option_jelze | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class DefaultParty(RandomGroupsChoice, TextChoice): | 
					
						
							|  |  |  |  |     """Preselect the party lineup.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     (Only has an effect if shuffle_party_members is set to false.) | 
					
						
							|  |  |  |  |     Supported values: | 
					
						
							|  |  |  |  |     Can be set to any valid combination of up to 4 party member initials, e.g.: | 
					
						
							|  |  |  |  |     M — Maxim | 
					
						
							|  |  |  |  |     DGMA — Dekar, Guy, Maxim, and Arty | 
					
						
							|  |  |  |  |     MSTL — Maxim, Selan, Tia, and Lexis | 
					
						
							|  |  |  |  |     random-2p — a random 2-person party | 
					
						
							|  |  |  |  |     random-3p — a random 3-person party | 
					
						
							|  |  |  |  |     random-4p — a random 4-person party | 
					
						
							|  |  |  |  |     Default value: M | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Default party lineup" | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  |     default: Union[str, int] = "M" | 
					
						
							|  |  |  |  |     value: Union[str, int] | 
					
						
							| 
									
										
										
										
											2022-12-18 14:44:20 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     random_groups = { | 
					
						
							|  |  |  |  |         "random-2p": ["M" + "".join(p) for p in combinations("ADGLST", 1)], | 
					
						
							|  |  |  |  |         "random-3p": ["M" + "".join(p) for p in combinations("ADGLST", 2)], | 
					
						
							|  |  |  |  |         "random-4p": ["M" + "".join(p) for p in combinations("ADGLST", 3)], | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     vars().update({f"option_{party}": party for party in (*random_groups, "M", *chain(*random_groups.values()))}) | 
					
						
							|  |  |  |  |     _valid_sorted_parties: List[List[str]] = [sorted(party) for party in ("M", *chain(*random_groups.values()))] | 
					
						
							|  |  |  |  |     _members_to_bytes: bytes = bytes.maketrans(b"MSGATDL", bytes(range(7))) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  |     def verify(self, world: Type[World], player_name: str, plando_options: PlandoOptions) -> None: | 
					
						
							| 
									
										
										
										
											2022-12-18 14:44:20 +01:00
										 |  |  |  |         if str(self.value).lower() in self.random_groups: | 
					
						
							|  |  |  |  |             return | 
					
						
							|  |  |  |  |         if sorted(str(self.value).upper()) in self._valid_sorted_parties: | 
					
						
							|  |  |  |  |             return | 
					
						
							|  |  |  |  |         raise ValueError(f"Could not find option '{self.value}' for '{self.__class__.__name__}', known options are:\n" | 
					
						
							|  |  |  |  |                          f"{', '.join(self.random_groups)}, {', '.join(('M', *chain(*self.random_groups.values())))} " | 
					
						
							|  |  |  |  |                          "as well as all permutations of these.") | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @staticmethod | 
					
						
							|  |  |  |  |     def _flip(i: int) -> int: | 
					
						
							|  |  |  |  |         return {4: 5, 5: 4}.get(i, i) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def event_script(self) -> bytes: | 
					
						
							|  |  |  |  |         return bytes((*(b for i in bytes(self) if i != 0 for b in (0x2B, i, 0x2E, i + 0x65, 0x1A, self._flip(i) + 1)), | 
					
						
							|  |  |  |  |                       0x1E, 0x0B, len(self) - 1, 0x1C, 0x86, 0x03, *(0x00,) * (6 * (4 - len(self))))) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def roster(self) -> bytes: | 
					
						
							|  |  |  |  |         return bytes((len(self), *bytes(self), *(0xFF,) * (4 - len(self)))) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __bytes__(self) -> bytes: | 
					
						
							|  |  |  |  |         return str(self.value).upper().encode("ASCII").translate(self._members_to_bytes) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __len__(self) -> int: | 
					
						
							|  |  |  |  |         return len(str(self.value)) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  | class EnemyFloorNumbers(Choice): | 
					
						
							|  |  |  |  |     """Change which enemy types are encountered at which floor numbers.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: | 
					
						
							|  |  |  |  |     vanilla | 
					
						
							|  |  |  |  |         Ninja, e.g., is allowed to appear on the 3 floors B44-B46 | 
					
						
							|  |  |  |  |     shuffle — The existing enemy types are redistributed among nearby floors. Shifts by up to 6 floors are possible. | 
					
						
							|  |  |  |  |         Ninja, e.g., will be allowed to appear on exactly 3 consecutive floors somewhere from B38-B40 to B50-B52 | 
					
						
							|  |  |  |  |     randomize — For each floor, new enemy types are chosen randomly from the set usually possible on floors [-6, +6]. | 
					
						
							|  |  |  |  |         Ninja, e.g., is among the various possible selections for any enemy slot affecting the floors from B38 to B52 | 
					
						
							|  |  |  |  |     Default value: vanilla (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Enemy floor numbers" | 
					
						
							|  |  |  |  |     option_vanilla = 0 | 
					
						
							|  |  |  |  |     option_shuffle = 1 | 
					
						
							|  |  |  |  |     option_randomize = 2 | 
					
						
							|  |  |  |  |     default = option_vanilla | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class EnemyMovementPatterns(EnemyChoice): | 
					
						
							|  |  |  |  |     """Change the movement patterns of enemies.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: | 
					
						
							|  |  |  |  |     vanilla | 
					
						
							|  |  |  |  |     shuffle_by_pattern — The existing movement patterns are redistributed among each other. | 
					
						
							|  |  |  |  |         Sprites that usually share a movement pattern will still share movement patterns after shuffling | 
					
						
							|  |  |  |  |     randomize_by_pattern — For each movement pattern, a new one is chosen randomly from the set of existing patterns. | 
					
						
							|  |  |  |  |         Sprites that usually share a movement pattern will still share movement patterns after randomizing | 
					
						
							|  |  |  |  |     shuffle_by_sprite — The existing movement patterns of sprites are redistributed among the enemy sprites. | 
					
						
							|  |  |  |  |         Sprites that usually share a movement pattern can end up with different movement patterns after shuffling | 
					
						
							|  |  |  |  |     randomize_by_sprite — For each sprite, a new movement is chosen randomly from the set of existing patterns. | 
					
						
							|  |  |  |  |         Sprites that usually share a movement pattern can end up with different movement patterns after randomizing | 
					
						
							|  |  |  |  |     singularity — All enemy sprites use the same, randomly selected movement pattern | 
					
						
							|  |  |  |  |     Alternatively, you can directly specify an enemy name such as "Red Jelly" as the value of this option. | 
					
						
							|  |  |  |  |         In that case, the movement pattern usually associated with this sprite will be used by all enemy sprites | 
					
						
							|  |  |  |  |     Default value: vanilla (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Enemy movement patterns" | 
					
						
							|  |  |  |  |     option_vanilla = 0 | 
					
						
							|  |  |  |  |     option_shuffle_by_pattern = 1 | 
					
						
							|  |  |  |  |     option_randomize_by_pattern = 2 | 
					
						
							|  |  |  |  |     option_shuffle_by_sprite = 3 | 
					
						
							|  |  |  |  |     option_randomize_by_sprite = 4 | 
					
						
							|  |  |  |  |     option_singularity = 5 | 
					
						
							|  |  |  |  |     default = option_vanilla | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class EnemySprites(EnemyChoice): | 
					
						
							|  |  |  |  |     """Change the appearance of enemies.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: | 
					
						
							|  |  |  |  |     vanilla | 
					
						
							|  |  |  |  |     shuffle — The existing sprites are redistributed among the enemy types. | 
					
						
							|  |  |  |  |         This means that, after shuffling, exactly 1 enemy type will be dressing up as the "Red Jelly" sprite | 
					
						
							|  |  |  |  |     randomize — For each enemy type, a new sprite is chosen randomly from the set of existing sprites. | 
					
						
							|  |  |  |  |         This means that, after randomizing, any number of enemy types could end up using the "Red Jelly" sprite | 
					
						
							|  |  |  |  |     singularity — All enemies use the same, randomly selected sprite | 
					
						
							|  |  |  |  |     Alternatively, you can directly specify an enemy name such as "Red Jelly" as the value of this option. | 
					
						
							|  |  |  |  |         In this case, the sprite usually associated with that enemy will be used by all enemies | 
					
						
							|  |  |  |  |     Default value: vanilla (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Enemy sprites" | 
					
						
							|  |  |  |  |     option_vanilla = 0 | 
					
						
							|  |  |  |  |     option_shuffle = 1 | 
					
						
							|  |  |  |  |     option_randomize = 2 | 
					
						
							|  |  |  |  |     option_singularity = 3 | 
					
						
							|  |  |  |  |     default = option_vanilla | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class ExpModifier(Range): | 
					
						
							|  |  |  |  |     """Percentage modifier for EXP gained from enemies.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: 100 – 500 | 
					
						
							|  |  |  |  |     Default value: 100 (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "EXP modifier" | 
					
						
							|  |  |  |  |     range_start = 100 | 
					
						
							|  |  |  |  |     range_end = 500 | 
					
						
							|  |  |  |  |     default = 100 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __call__(self, exp: bytes) -> bytes: | 
					
						
							|  |  |  |  |         try: | 
					
						
							|  |  |  |  |             return (int.from_bytes(exp, "little") * self.value // 100).to_bytes(2, "little") | 
					
						
							|  |  |  |  |         except OverflowError: | 
					
						
							|  |  |  |  |             return b"\xFF\xFF" | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  | class FinalFloor(Range): | 
					
						
							|  |  |  |  |     """The final floor, where the boss resides.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: 2 – 99 | 
					
						
							|  |  |  |  |     Default value: 99 (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Final floor" | 
					
						
							|  |  |  |  |     range_start = 2 | 
					
						
							|  |  |  |  |     range_end = 99 | 
					
						
							|  |  |  |  |     default = 99 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class GearVarietyAfterB9(Toggle): | 
					
						
							|  |  |  |  |     """Fixes a bug that prevents various gear from appearing after B9.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Due to an overflow bug in the game, the distribution of red chest gear is impaired after B9. | 
					
						
							|  |  |  |  |     Starting with B10, the number of items available from red chests is severely limited, meaning that red chests will | 
					
						
							|  |  |  |  |     no longer contain any shields, headgear, rings, or jewels (and the selection of body armor is reduced as well). | 
					
						
							|  |  |  |  |     This setting provides a fix for the bug described above. | 
					
						
							|  |  |  |  |     If enabled, red chests beyond B9 will continue to produce shields, headgear, rings, and jewels as intended, | 
					
						
							|  |  |  |  |     while the odds of finding body armor in red chests are decreased as a result. | 
					
						
							|  |  |  |  |     The distributions of red chest weapons, spells, and consumables as well as blue chests are unaffected. | 
					
						
							|  |  |  |  |     Supported values: false, true | 
					
						
							|  |  |  |  |     Default value: false (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Increase gear variety after B9" | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class Goal(Choice): | 
					
						
							|  |  |  |  |     """The objective you have to fulfill in order to complete the game.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: | 
					
						
							|  |  |  |  |     boss — defeat the boss on the final floor | 
					
						
							|  |  |  |  |     iris_treasure_hunt — gather the required number of Iris treasures and leave the cave | 
					
						
							|  |  |  |  |     boss_iris_treasure_hunt — complete both the "boss" and the "iris_treasure_hunt" objective (in any order) | 
					
						
							|  |  |  |  |     final_floor — merely reach the final floor | 
					
						
							|  |  |  |  |     Default value: boss | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Goal" | 
					
						
							|  |  |  |  |     option_boss = 0x01 | 
					
						
							|  |  |  |  |     option_iris_treasure_hunt = 0x02 | 
					
						
							|  |  |  |  |     option_boss_iris_treasure_hunt = 0x03 | 
					
						
							|  |  |  |  |     option_final_floor = 0x04 | 
					
						
							|  |  |  |  |     default = option_boss | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-21 23:27:30 +02:00
										 |  |  |  | class GoldModifier(Range): | 
					
						
							|  |  |  |  |     """Percentage modifier for gold gained from enemies.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: 25 – 400 | 
					
						
							|  |  |  |  |     Default value: 100 (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Gold modifier" | 
					
						
							|  |  |  |  |     range_start = 25 | 
					
						
							|  |  |  |  |     range_end = 400 | 
					
						
							|  |  |  |  |     default = 100 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __call__(self, gold: bytes) -> bytes: | 
					
						
							|  |  |  |  |         try: | 
					
						
							|  |  |  |  |             return (int.from_bytes(gold, "little") * self.value // 100).to_bytes(2, "little") | 
					
						
							|  |  |  |  |         except OverflowError: | 
					
						
							|  |  |  |  |             return b"\xFF\xFF" | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  | class HealingFloorChance(Range): | 
					
						
							|  |  |  |  |     """The chance of a floor having a healing tile hidden under a bush.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     It is given in units of 1/256, i.e., a value of 16 corresponds to 16/256 = 6.25%. | 
					
						
							|  |  |  |  |     Supported values: 0 – 255 | 
					
						
							|  |  |  |  |     Default value: 16 (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Healing tile floor chance" | 
					
						
							|  |  |  |  |     range_start = 0 | 
					
						
							|  |  |  |  |     range_end = 255 | 
					
						
							|  |  |  |  |     default = 16 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class InitialFloor(Range): | 
					
						
							|  |  |  |  |     """The initial floor, where you begin your journey.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     (If this value isn't smaller than the value of final_floor, it will automatically be set to final_floor - 1) | 
					
						
							|  |  |  |  |     Supported values: 1 – 98 | 
					
						
							|  |  |  |  |     Default value: 1 (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Initial floor" | 
					
						
							|  |  |  |  |     range_start = 1 | 
					
						
							|  |  |  |  |     range_end = 98 | 
					
						
							|  |  |  |  |     default = 1 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class IrisFloorChance(Range): | 
					
						
							|  |  |  |  |     """The chance of a floor being able to generate an Iris treasure.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     It is given in units of 1/256, i.e., a value of 5 corresponds to 5/256 ~ 1.95%. | 
					
						
							|  |  |  |  |     The true chance of a floor holding an Iris treasure you need is usually lower than the chance specified here, e.g., | 
					
						
							|  |  |  |  |     if you have already found 8 of 9 Iris items then the chance of generating the last one is only 1/9 of this value. | 
					
						
							|  |  |  |  |     Supported values: 5 – 255 | 
					
						
							|  |  |  |  |     Default value: 5 (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Iris treasure floor chance" | 
					
						
							|  |  |  |  |     range_start = 5 | 
					
						
							|  |  |  |  |     range_end = 255 | 
					
						
							|  |  |  |  |     default = 5 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class IrisTreasuresRequired(Range): | 
					
						
							|  |  |  |  |     """The number of Iris treasures required to complete the goal.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     This setting only has an effect if the "iris_treasure_hunt" or "boss_iris_treasure_hunt" goal is active. | 
					
						
							|  |  |  |  |     Supported values: 1 – 9 | 
					
						
							|  |  |  |  |     Default value: 9 | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Iris treasures required" | 
					
						
							|  |  |  |  |     range_start = 1 | 
					
						
							|  |  |  |  |     range_end = 9 | 
					
						
							|  |  |  |  |     default = 9 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  | class MasterHp(Range): | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  |     """The number of hit points of the Master
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  |     (Only has an effect if boss is set to master.) | 
					
						
							|  |  |  |  |     Supported values: 1 – 9980 | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  |     Default value: 9980 (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Master HP" | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  |     range_start = 1 | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  |     range_end = 9980 | 
					
						
							|  |  |  |  |     default = 9980 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class PartyStartingLevel(LevelMixin, Range): | 
					
						
							|  |  |  |  |     """The starting level of your party members.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: 1 – 99 | 
					
						
							|  |  |  |  |     Default value: 1 (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Party starting level" | 
					
						
							|  |  |  |  |     range_start = 1 | 
					
						
							|  |  |  |  |     range_end = 99 | 
					
						
							|  |  |  |  |     default = 1 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def xp(self) -> int: | 
					
						
							|  |  |  |  |         return self._to_xp(self.value, capsule=False) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class RunSpeed(Choice): | 
					
						
							|  |  |  |  |     """Modifies the game to allow you to move faster than normal when pressing the Y button.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: disabled, double, triple, quadruple | 
					
						
							|  |  |  |  |     Default value: disabled (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Run speed" | 
					
						
							|  |  |  |  |     option_disabled = 0x08 | 
					
						
							|  |  |  |  |     option_double = 0x10 | 
					
						
							|  |  |  |  |     option_triple = 0x16 | 
					
						
							|  |  |  |  |     option_quadruple = 0x20 | 
					
						
							|  |  |  |  |     default = option_disabled | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-21 23:27:30 +02:00
										 |  |  |  | class ShopInterval(SpecialRange): | 
					
						
							|  |  |  |  |     """Place shops after a certain number of floors.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     E.g., if you set this to 5, then you will be given the opportunity to shop after completing B5, B10, B15, etc., | 
					
						
							|  |  |  |  |     whereas if you set it to 1, then there will be a shop after every single completed floor. | 
					
						
							|  |  |  |  |     Shops will offer a random selection of wares; on deeper floors, more expensive items might appear. | 
					
						
							|  |  |  |  |     You can customize the stock that can appear in shops using the shop_inventory option. | 
					
						
							|  |  |  |  |     You can control how much gold you will be obtaining from enemies using the gold_multiplier option. | 
					
						
							|  |  |  |  |     Supported values: disabled, 1 – 10 | 
					
						
							|  |  |  |  |     Default value: disabled (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Shop interval" | 
					
						
							|  |  |  |  |     range_start = 0 | 
					
						
							|  |  |  |  |     range_end = 10 | 
					
						
							|  |  |  |  |     default = 0 | 
					
						
							|  |  |  |  |     special_range_cutoff = 1 | 
					
						
							|  |  |  |  |     special_range_names = { | 
					
						
							|  |  |  |  |         "disabled": 0, | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class ShopInventory(OptionDict): | 
					
						
							|  |  |  |  |     """Determine the item types that can appear in shops.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     The value of this option should be a mapping of item categories (or individual items) to weights (non-negative | 
					
						
							|  |  |  |  |     integers), which are used as relative probabilities when it comes to including these things in shops. (The actual | 
					
						
							|  |  |  |  |     contents of the generated shops are selected randomly and are subject to additional constraints such as more | 
					
						
							|  |  |  |  |     expensive things being allowed only on later floors.) | 
					
						
							|  |  |  |  |     Supported keys: | 
					
						
							|  |  |  |  |     non_restorative — a selection of mostly non-restorative red chest consumables | 
					
						
							|  |  |  |  |     restorative — all HP- or MP-restoring red chest consumables | 
					
						
							|  |  |  |  |     blue_chest — all blue chest items | 
					
						
							|  |  |  |  |     spell — all red chest spells | 
					
						
							|  |  |  |  |     gear — all red chest armors, shields, headgear, rings, and rocks (this respects the gear_variety_after_b9 option, | 
					
						
							|  |  |  |  |         meaning that you will not encounter any shields, headgear, rings, or rocks in shops from B10 onward unless you | 
					
						
							|  |  |  |  |         also enabled that option) | 
					
						
							|  |  |  |  |     weapon — all red chest weapons | 
					
						
							|  |  |  |  |     Additionally, you can also add extra weights for any specific cave item. If you want your shops to have a | 
					
						
							|  |  |  |  |     higher than normal chance of selling a Dekar blade, you can, e.g., add "Dekar blade: 5". | 
					
						
							|  |  |  |  |     You can even forego the predefined categories entirely and design a custom shop pool from scratch by providing | 
					
						
							|  |  |  |  |     separate weights for each item you want to include. | 
					
						
							|  |  |  |  |     (Spells, however, cannot be weighted individually and are only available as part of the "spell" category.) | 
					
						
							|  |  |  |  |     Default value: {spell: 30, gear: 45, weapon: 82} | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Shop inventory" | 
					
						
							|  |  |  |  |     _special_keys = {"non_restorative", "restorative", "blue_chest", "spell", "gear", "weapon"} | 
					
						
							|  |  |  |  |     valid_keys = _special_keys | {item for item, data in l2ac_item_table.items() | 
					
						
							|  |  |  |  |                                   if data.type in {ItemType.BLUE_CHEST, ItemType.ENEMY_DROP, ItemType.ENTRANCE_CHEST, | 
					
						
							|  |  |  |  |                                                    ItemType.RED_CHEST, ItemType.RED_CHEST_PATCH}} | 
					
						
							|  |  |  |  |     default: Dict[str, int] = { | 
					
						
							|  |  |  |  |         "spell": 30, | 
					
						
							|  |  |  |  |         "gear": 45, | 
					
						
							|  |  |  |  |         "weapon": 82, | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     value: Dict[str, int] | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def verify(self, world: Type[World], player_name: str, plando_options: PlandoOptions) -> None: | 
					
						
							|  |  |  |  |         super().verify(world, player_name, plando_options) | 
					
						
							|  |  |  |  |         for item, weight in self.value.items(): | 
					
						
							|  |  |  |  |             if not isinstance(weight, numbers.Integral) or weight < 0: | 
					
						
							|  |  |  |  |                 raise Exception(f"Weight for item \"{item}\" from option {self} must be a non-negative integer, " | 
					
						
							|  |  |  |  |                                 f"but was \"{weight}\".") | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def total(self) -> int: | 
					
						
							|  |  |  |  |         return sum(self.value.values()) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def non_restorative(self) -> int: | 
					
						
							|  |  |  |  |         return self.value.get("non_restorative", 0) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def restorative(self) -> int: | 
					
						
							|  |  |  |  |         return self.value.get("restorative", 0) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def blue_chest(self) -> int: | 
					
						
							|  |  |  |  |         return self.value.get("blue_chest", 0) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def spell(self) -> int: | 
					
						
							|  |  |  |  |         return self.value.get("spell", 0) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def gear(self) -> int: | 
					
						
							|  |  |  |  |         return self.value.get("gear", 0) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def weapon(self) -> int: | 
					
						
							|  |  |  |  |         return self.value.get("weapon", 0) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @functools.cached_property | 
					
						
							|  |  |  |  |     def custom(self) -> Dict[int, int]: | 
					
						
							|  |  |  |  |         return {l2ac_item_table[item].code & 0x01FF: weight for item, weight in self.value.items() | 
					
						
							|  |  |  |  |                 if item not in self._special_keys} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  | class ShuffleCapsuleMonsters(Toggle): | 
					
						
							|  |  |  |  |     """Shuffle the capsule monsters into the multiworld.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: | 
					
						
							|  |  |  |  |     false — all 7 capsule monsters are available in the menu and can be selected right away | 
					
						
							|  |  |  |  |     true — you start without capsule monster; 7 new "items" are added to your pool and shuffled into the multiworld; | 
					
						
							|  |  |  |  |         when one of these items is found, the corresponding capsule monster is unlocked for you to use | 
					
						
							|  |  |  |  |     Default value: false (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Shuffle capsule monsters" | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def unlock(self) -> int: | 
					
						
							|  |  |  |  |         return 0b00000000 if self.value else 0b01111111 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | class ShufflePartyMembers(Toggle): | 
					
						
							|  |  |  |  |     """Shuffle the party members into the multiworld.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     Supported values: | 
					
						
							|  |  |  |  |     false — all 6 optional party members are present in the cafe and can be recruited right away | 
					
						
							|  |  |  |  |     true — only Maxim is available from the start; 6 new "items" are added to your pool and shuffled into the | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  |         multiworld; when one of these items is found, the corresponding party member is unlocked for you to use. | 
					
						
							|  |  |  |  |         While cave diving, you can add newly unlocked ones to your party by using the character items from the inventory | 
					
						
							| 
									
										
										
										
											2022-12-12 02:36:18 +01:00
										 |  |  |  |     Default value: false (same as in an unmodified game) | 
					
						
							|  |  |  |  |     """
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     display_name = "Shuffle party members" | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     @property | 
					
						
							|  |  |  |  |     def unlock(self) -> int: | 
					
						
							|  |  |  |  |         return 0b00000000 if self.value else 0b11111100 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  | @dataclass | 
					
						
							| 
									
										
										
										
											2023-10-15 04:53:28 +02:00
										 |  |  |  | class L2ACOptions(PerGameCommonOptions): | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  |     blue_chest_chance: BlueChestChance | 
					
						
							|  |  |  |  |     blue_chest_count: BlueChestCount | 
					
						
							|  |  |  |  |     boss: Boss | 
					
						
							|  |  |  |  |     capsule_cravings_jp_style: CapsuleCravingsJPStyle | 
					
						
							|  |  |  |  |     capsule_starting_form: CapsuleStartingForm | 
					
						
							|  |  |  |  |     capsule_starting_level: CapsuleStartingLevel | 
					
						
							|  |  |  |  |     crowded_floor_chance: CrowdedFloorChance | 
					
						
							|  |  |  |  |     custom_item_pool: CustomItemPool | 
					
						
							|  |  |  |  |     death_link: DeathLink | 
					
						
							|  |  |  |  |     default_capsule: DefaultCapsule | 
					
						
							|  |  |  |  |     default_party: DefaultParty | 
					
						
							|  |  |  |  |     enemy_floor_numbers: EnemyFloorNumbers | 
					
						
							|  |  |  |  |     enemy_movement_patterns: EnemyMovementPatterns | 
					
						
							|  |  |  |  |     enemy_sprites: EnemySprites | 
					
						
							|  |  |  |  |     exp_modifier: ExpModifier | 
					
						
							|  |  |  |  |     final_floor: FinalFloor | 
					
						
							|  |  |  |  |     gear_variety_after_b9: GearVarietyAfterB9 | 
					
						
							|  |  |  |  |     goal: Goal | 
					
						
							| 
									
										
										
										
											2023-10-21 23:27:30 +02:00
										 |  |  |  |     gold_modifier: GoldModifier | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  |     healing_floor_chance: HealingFloorChance | 
					
						
							|  |  |  |  |     initial_floor: InitialFloor | 
					
						
							|  |  |  |  |     iris_floor_chance: IrisFloorChance | 
					
						
							|  |  |  |  |     iris_treasures_required: IrisTreasuresRequired | 
					
						
							|  |  |  |  |     master_hp: MasterHp | 
					
						
							|  |  |  |  |     party_starting_level: PartyStartingLevel | 
					
						
							|  |  |  |  |     run_speed: RunSpeed | 
					
						
							| 
									
										
										
										
											2023-10-21 23:27:30 +02:00
										 |  |  |  |     shop_interval: ShopInterval | 
					
						
							|  |  |  |  |     shop_inventory: ShopInventory | 
					
						
							| 
									
										
											  
											
												lufia2ac: new features, bug fixes, and more (#1549)
### New features
- ***Architect mode***
  Usually the cave is randomized by the game, meaning that each attempt will produce a different dungeon. However, with this new feature the player can, between runs, opt into keeping the same cave. If activated, they will then encounter the same floor layouts, same enemy spawns, and same red chest contents as on their previous attempt.   
- ***Custom item pool***
  Previously, the multiworld item pool consisted entirely of random blue chest items because, well, the permanent checks are blue chests and that's what one would normally get from these. While blue chest items often greatly increase your odds against regular enemies, being able to defeat the Master can be contingent on having an appropriate equipment setup of red chest items (such as Dekar blade) or even enemy drops (such as Hidora rock), most of which cannot normally be obtained from blue chests.
  With the custom item pool option, players now have the freedom to place any cave item into the multiworld itempool for their world.
- ***Enemy floor number, enemy sprite, and enemy movement pattern randomization***
  Experienced players can deduce a lot of information about the opposition they will be facing, for example: Given the current floor number, one can know in advance which of the enemy types will have a chance to spawn on that floor. And when seeing a particular enemy sprite, one can already know which enemy types one might have to face in battle if one were to come in contact with it, and also how that enemy group will move through the dungeon.
  Three new randomization options are added for players who want to spice up their game: one can shuffle which enemy types appear on which floor, one can shuffle which sprite is used by which enemy type, and one can shuffle which movement pattern is used by which sprite.
- ***EXP modifier***
  Just a simple multiplier option to allow people to level up faster. (For technical reasons, the maximum amount of EXP that can be awarded for a single enemy is limited to 65535, but even with the maximum allowed modifier of 500% there are only 6 enemy types in the cave that can reach this cap.)
### Balance change
- ***proportionally adjust chest type distribution to accommodate increased blue chest chance***
  One of the main problems that became apparent in the current version has to do with the distribution of chest contents. The game considers 6 categories, namely: consumable (mostly non-restorative), consumable (restorative), blue chest item, spell, gear, and weapon. Since only blue chests count as multiworld locations, we want to have a mechanism to customize the blue chest chance.
  Given how the chest types are detetermined in game, a naive implementation of an increased blue chest chance causes only the consumable chance to be decreased in return. In practice, this has resulted in some players of worlds with a high blue chest chance struggling (more than usual) to keep their party alive because they were always low on comsumables that restore HP and MP.
  The new algorithm tries to avoid this one-sided effect by having an increase in blue chest chance resulting in a decrease of all other types, calculated in such a way that the relative distribution of the other 5 categories stays (approximately) the same.
### Bug fixes
- ***prevent using party member items if character is already in party***
  This should have been changed at the same time that 6eb00621e39c930f5746f5f3c69a6bc19cd0e84a was made, but oh well... 
- ***fix glitched sprite when opening a chest immediately after receiving an item***
  When opening a chest right after receiving a multiworld item (such that there were two item get animations in the exact same iteration of the game main loop), the item from the chest would display an incorrect sprite in the wrong place. Fixed by cleaning up some relevant memory addresses after getting the multiworld item.
- ***fix death link***
  There was a condition in `deathlink_kill_player` that looked kinda smart (it checked the time against `last_death_link`), but actually wasn't smart at all because `deathlink_kill_player` is executed as an async task and the main thread will update `last_death_link` after creating the task, meaning that whether or not the incoming death link would actually be passed to the game seems to have been up to a race condition. Fixed by simply removing that check.
### Other
- ***add Lufia II Ancient Cave (and SMW) to the network diagram***
  These two games were missing from the SNES sector.
- ***implement get_filler_item_name***
  Place a restorative consumable instead of a completely random item. (Now the only known problem with item links in lufia2ac is... that noone has ever tested item links. But this should be an improvement at least. Anyway, now #1172 can come ;)
  And btw., if you think that the implementation of random selection in this method looks weird, that's because it is indeed weird. (It tries to recreate the algorithm that the game itself uses when it generates a replacement item for a chest that would contain a spell that the party already knows.)
- ***store all options in a dataclass***
  This is basically like using #993 (but without actual support from core). It makes the lufia2ac world code much nicer to maintain because one doesn't have to change 5 different places anymore when adding or renaming an option.
- ***remove master_hp.scale***
  I have to admit: `scale` was a mistake. Never have I seen a single option value cause so many user misconceptions. Some people assume it affects enemies other than the Master; some people assume it affects stats other than HP; and many people will just assume it is a magic option that will somehow counterbalance whatever settings combination they are currently trying to shoot themselves in the foot with.
  On top of that, the `scale` mechanism probably doesn't provide a good user experience even when used for its intended purpose (since having reached floor XY in general doesn't mean you will have the power to deplete XY% of the Masters usual HP; especially given that, due to the randomness of loot, you are never guaranteed to be able to defeat the vanilla Master even when you have cleared 100% of the floors).
  The intended target audience of the `master_hp` option are people who want to fight the Master (and know how to fight it), but also want to lessen (to a degree of their choosing) the harsh dependence on the specific equipment setups that are usually required to win this fight even when having done all 99 floors. They can achieve this by setting the `master_hp` option to a numeric value appropriate for the level of challenge they are seeking. Therefore, nothing of value should be lost by removing the special `scale` value from the `master_hp` option, while at the same time a major source of user confusion will be eliminated.
- ***typing***
  This (combined with the switch to the option dataclass) greatly reduces the typing problems in the lufia2ac world. The remaining typing errors mostly fall into 4 categories:
  1. Lambdas with defaults (which seem to be incorrectly reported as an error due to a mypy bug)
  1. Classmethods that return instances (which could probably be improved using PEP 673 "Self" types, but that would require Python 3.11 as the minimum supported version)
  1. Everything that inherits from TextChoice (which is a typing mess in core)
  1. Everything related to asar.py (which does not have proper typing and lies outside of this project)
## How was this tested?
https://discord.com/channels/731205301247803413/1080852357442707476 and others
											
										 
											2023-03-20 17:04:57 +01:00
										 |  |  |  |     shuffle_capsule_monsters: ShuffleCapsuleMonsters | 
					
						
							|  |  |  |  |     shuffle_party_members: ShufflePartyMembers |