Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
Sponge | SpongeAPI
This PR attempts to solve long standing issues with the attack and damage events. The previous API was attempting to represent the damage calculation as a chain of mathematical operations which can be modified or replayed with a different base damage. This was problematic because it implies the implementation must be capable to capture all operations to be able to replay. However, due to the nature of modded platforms, it is impossible to capture everything as a mathematical operation. This results in a incorrect final damage value and thus angry pvp players.
Let's take an example: the shield modifier. Modded platforms have a shield event where mods can just set the current damage value. Was it an addition, a multiplication or any other mathematical operation? We can't know. We could just capture it as a constant, but being able to replay with a different base damage becomes useless. We could instead capture a modifier that would re-fire the event each time we replay the chain, however this would certainly break mods that have side-effects.
Another problem is that we represent the damage calculation as a chain. First of all, it is not. It has branches, early returns, splits and merges. Even if it was a chain, a mod could do some bytecode modification and add new operations in the middle of the "chain". Since we can't capture these new operations, our representation of the chain will be incomplete.
Finally, due to having a single event firing after the whole damage calculation is done, the implementation has to cancel all side-effects during the damage calculation, then replay them after the event. Again, this will not work with mod side-effects but even for vanilla this is a brittle and cumbersome approach
Note that even if it seems that these events were mostly problematic on modded platforms (indeed they were completely broken), I have also seen a few mistakes in the vanilla implementation, especially for the invulnerability time mechanic. This is because the API can hardly represent this mechanic correctly.
Now let's talk about the proposed solution.
Forget about the previous concept of
DamageModifier
and let's introduce the concept ofDamageStep
which represent a step in the damage calculation that we want to expose to the API. It no longer captures the mathematical operation, but just the values before and after the step. (consequence of 1.)These steps are just some important key points in the damage calculation, there are not an exhaustive representation of all the things that may happen during the damage calculation. (consequence of 2.)
Now, we split the damage event into two parts: Pre and Post.
Pre happens before the damage calculation and all its side-effects, at this time, we only know the base damage.
Post happens after the damage calculation, all steps have been captured.
During the Pre event, plugins can add some callbacks that will modify the damage value just before or just after a step. These callbacks are called
DamageModifier
. These callbacks are executed after the Pre event, in the right place during the damage calculation, ensuring that mods and side-effects will adapt to the modified value. (consequence of 3.).As a bonus, the attack and damage events now share most of their logic in a parent class.