org.spongepowered.api.event.entity.AttackEntityEvent Maven / Gradle / Ivy
/*
* This file is part of SpongeAPI, licensed under the MIT License (MIT).
*
* Copyright (c) SpongePowered
* Copyright (c) contributors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package org.spongepowered.api.event.entity;
import org.spongepowered.api.block.entity.carrier.Dispenser;
import org.spongepowered.api.effect.potion.PotionEffect;
import org.spongepowered.api.entity.Entity;
import org.spongepowered.api.entity.living.Living;
import org.spongepowered.api.entity.living.monster.skeleton.Skeleton;
import org.spongepowered.api.entity.living.player.Player;
import org.spongepowered.api.entity.projectile.arrow.ArrowLike;
import org.spongepowered.api.event.Cancellable;
import org.spongepowered.api.event.Cause;
import org.spongepowered.api.event.Event;
import org.spongepowered.api.event.cause.entity.damage.DamageFunction;
import org.spongepowered.api.event.cause.entity.damage.DamageModifier;
import org.spongepowered.api.event.cause.entity.damage.DamageModifierType;
import org.spongepowered.api.event.cause.entity.damage.DamageModifierTypes;
import org.spongepowered.api.event.cause.entity.damage.DamageType;
import org.spongepowered.api.event.cause.entity.damage.source.DamageSource;
import org.spongepowered.api.event.impl.entity.AbstractAttackEntityEvent;
import org.spongepowered.api.item.ItemTypes;
import org.spongepowered.api.item.inventory.ItemStack;
import org.spongepowered.api.item.inventory.ItemStackSnapshot;
import org.spongepowered.api.util.Tuple;
import org.spongepowered.api.util.annotation.eventgen.ImplementedBy;
import org.spongepowered.api.util.annotation.eventgen.PropertySettings;
import org.spongepowered.api.world.World;
import org.spongepowered.api.world.difficulty.Difficulties;
import org.spongepowered.api.world.difficulty.Difficulty;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.function.DoubleUnaryOperator;
import java.util.function.Function;
/**
* Represents the base event for when an {@link Entity} is being "attacked".
* The uniqueness of this event is that all {@link DamageSource}s can deal
* varying amounts of damage with varying modifiers based on various reasons.
* Due to this ambiguous variety of information that is possible to provide,
* the {@link AttackEntityEvent} can be summarized as so:
*
* An {@link ArrowLike},
* that was shot by a {@link Skeleton},
* with an enchanted {@link ItemTypes#BOW} {@link ItemStack},
* when the {@link World} {@link Difficulty} was set to
* {@link Difficulties#HARD},
* will deal possibly "5" damage to any {@link Entity} it hits.
*
*
* The issue with representing this type of "logic flow" is that a
* particular amount of damage from a {@link DamageSource}, even if specified
* to a particular {@link DamageType}, has no static finalized amount of damage
* to deal to a particular {@link Entity}. To properly represent this,
* a {@link DamageSource} has various "states" such as:
* {@link DamageSource#isAbsolute()}, or {@link DamageSource#isBypassingArmor()}.
* Quite simply, the {@link DamageSource} will always be the "first" element
* within a {@link Cause} that can be retrieved from {@link #cause()}.
*
* Next, any additional "aides" in attacking the {@link Entity} will
* be included in order of "priority of relation" to "attacking" the entity. In
* short, if another {@link Entity} is considered a "team member" to the
* attacking {@link Entity}, the "team member" may be a second element within
* the {@link Cause}. The same can be said if an {@link ArrowLike} was shot from
* a {@link Dispenser} that was triggered by a {@link Player} flipping a
* switch.
*
* Continuing with the notion of "modifiers" to damage, the "base" damage
* is modified or added onto after various unknown methods are called or
* processed on the damage. Optimally, these modifiers can be traced to a
* particular object, be it an {@link ItemStack}, {@link Difficulty}, or
* simply an an attribute. The interesting part is that these "modifiers"
* do not just define a static value to add to the "base" damage, they
* are usually a loose form of a {@link Function} that are applied to
* the "base" damage. Given that {@link Cause} has a unique capability of
* storing any and every {@link Object} willing to be passed into it, we
* can easily represent these "sources" of "modifiers" in a {@link Cause}.
* Now, knowing the "source" will not provide enough information, so a
* {@link DamageModifierType} is provided with a {@link DamageModifier} to
* paint the fullest picture of "explaining" the {@link DamageModifier} as to
* why it is present, and why it is "modifying" the "base" damage. Finally,
* we can associate a {@link DamageModifier} with a {@link Function} that is
* passed the current "damage" into {@link Function#apply(Object)}, being added
* to the current "damage". After all {@link DamageModifier} {@link Function}s
* are "applied", the overall "damage" is now the final damage to actually
* throw a {@link AttackEntityEvent}.
*
* Note that due to the mechanics of the game, {@link DamageModifier}s
* are always ordered in the order of which they apply their modifier onto
* the "base" damage. The implementation for {@link #finalOutputDamage()} can
* be exemplified like so:
*
* {@code
* double damage = this.baseDamage;
* for (Map.Entry> entry : this.modifierFunctions.entrySet()) {
* damage += checkNotNull(entry.getValue().apply(damage));
* }
* return damage;
* }
*
* TODO explain groups
* After which, the "final" damage is simply the summation of the
* "base" damage and all "modified damage" for each {@link DamageModifier}
* provided in this event.
*
* Coming forward, it is possible to further customize not only the "base"
* damage, but override pre-existing {@link DamageModifier} {@link Function}s
* by calling {@link #setOutputDamage(DamageModifier, DoubleUnaryOperator)} at which point the
* end result may be undefined. However, if a custom {@link DamageModifier}
* that aims to alter the "final" damage based on some custom circumstances,
* calling {@link #setOutputDamage(DamageModifier, DoubleUnaryOperator)} on a new
* {@link DamageModifier} instance, easily created from the
* {@link org.spongepowered.api.event.cause.entity.damage.DamageModifier.Builder},
* the provided pairing will be added at the
* "end" of the list for "modifying" the "base" damage.
*
* Note that this event is intended for processing incoming damage to
* an {@link Entity} prior to any {@link DamageModifier}s associated with
* the {@link #entity()}. The {@link AttackEntityEvent} is used
* to process the various {@link DamageModifier}s of which originate or are
* associated with the targeted {@link Entity}.
*/
@ImplementedBy(AbstractAttackEntityEvent.class)
public interface AttackEntityEvent extends Event, Cancellable {
/**
* For use with the {@link DamageSource} that is known as the "source"
* of the damage.
*/
String SOURCE = "Source";
/**
* For use with a {@link DamageModifier} where it's type is a
* {@link DamageModifierTypes#HARD_HAT} and the {@link Cause} contains
* an {@link ItemStackSnapshot}, usually a helmet.
*/
String HARD_HAT_ARMOR = "HardHat";
/**
* or use with a {@link DamageModifier} where its type is a
* {@link DamageModifierTypes#SHIELD} and the {@link Cause} contains
* an {@link ItemStackSnapshot} (in Vanilla, a shield).
*/
String SHIELD = "Shield";
/**
* For use with a {@link DamageModifier} where it's type is a
* {@link DamageModifierTypes#ARMOR} and the {@link Cause} contains
* an {@link ItemStackSnapshot}. Separate from hard hat but still
* considered as "armor" where it will absorb a certain amount of damage
* before dealing damage to the wearer.
*/
String GENERAL_ARMOR = "GeneralArmor";
/**
* For use with a {@link DamageModifier} where it's type is a
* {@link DamageModifierTypes#ARMOR} and the {@link Cause} contains
* an {@link ItemStackSnapshot} for a "helmet".
*/
String HELMET = AttackEntityEvent.GENERAL_ARMOR + ":head";
/**
* For use with a {@link DamageModifier} where it's type is a
* {@link DamageModifierTypes#ARMOR} and the {@link Cause} contains
* an {@link ItemStackSnapshot} for a "chestplate".
*/
String CHESTPLATE = AttackEntityEvent.GENERAL_ARMOR + ":chestplate";
/**
* For use with a {@link DamageModifier} where it's type is a
* {@link DamageModifierTypes#ARMOR} and the {@link Cause} contains
* an {@link ItemStackSnapshot} for "leggings".
*/
String LEGGINGS = AttackEntityEvent.GENERAL_ARMOR + ":leggings";
/**
* For use with a {@link DamageModifier} where it's type is a
* {@link DamageModifierTypes#ARMOR} and the {@link Cause} contains
* an {@link ItemStackSnapshot} for "boots".
*/
String BOOTS = AttackEntityEvent.GENERAL_ARMOR + ":boots";
/**
* For use with a {@link DamageModifier} where it's type is a
* {@link DamageModifierTypes#HARD_HAT} and the {@link Cause} contains
* a {@link PotionEffect}.
*/
String RESISTANCE = "Resistance";
/**
* For use with a {@link DamageModifier} where it's type is a
* {@link DamageModifierTypes#ABSORPTION} and the {@link Cause} contains
* a {@link PotionEffect}.
*/
String ABSORPTION = "AbsorptionPotion";
/**
* For use with a {@link DamageModifier} where the root cause is "created"
* by an object, usually the {@link Entity} or {@link Living} entity.
*/
String CREATOR = "Creator";
/**
* For use with a {@link DamageSource} with a {@link DamageSource#source() entity} or
* {@link DamageSource#blockSnapshot()} such that it was last "notified" by the object
* represented in the cause.
*
* Usually this is used where a {@link Player} interacted with the
* now {@link DamageSource} such that they
*/
String NOTIFIER = "Notifier";
/**
* Gets the {@link Entity}.
*
* @return The entity
*/
Entity entity();
/**
* Gets the original "raw" amount of damage to deal to the targeted
* {@link Entity}.
*
* @return The original "raw" damage
*/
double originalDamage();
/**
* Gets the original "final" amount of damage after all original
* {@link DamageModifier}s are applied to {@link #originalDamage()}.
* The "final" damage is considered the amount of health being lost by
* the {@link Entity}, if health is tracked.
*
* @return The final amount of damage to originally deal
*/
@PropertySettings(requiredParameter = false, generateMethods = false)
double originalFinalDamage();
/**
* Gets an immutable {@link Map} of all original {@link DamageModifier}s
* and their associated "modified" damage. Note that ordering is not
* retained.
*
* @return An immutable map of the original modified damages
*/
@PropertySettings(requiredParameter = false, generateMethods = false)
Map> originalDamages();
/**
* Gets the original damage for the provided {@link DamageModifier}. If
* the provided {@link DamageModifier} was not included in
* {@link #originalDamages()}, an {@link IllegalArgumentException} is
* thrown.
*
* @param damageModifier The original damage modifier
* @return The original damage change
*/
Tuple originalModifierDamage(DamageModifier damageModifier);
/**
* Gets the original {@link List} of {@link DamageModifier} to
* {@link Function} that was originally passed into the event.
*
* @return The list of damage modifier functions
*/
List originalFunctions();
/**
* Gets the "base" damage to deal to the targeted {@link Entity}. The
* "base" damage is the original value before passing along the chain of
* {@link Function}s for all known {@link DamageModifier}s.
*
* @return The base damage
*/
@PropertySettings(requiredParameter = false, generateMethods = false)
double baseOutputDamage();
/**
* Sets the "base" damage to deal to the targeted {@link Entity}. The
* "base" damage is the original value before passing along the chain of
* {@link Function}s for all known {@link DamageModifier}s.
*
* @param baseDamage The base damage
*/
void setBaseOutputDamage(double baseDamage);
/**
* Gets the final damage that will be passed into the proceeding
* {@link AttackEntityEvent}. The final damage is the end result of the
* {@link #baseOutputDamage()} being applied in {@link Function#apply(Object)}
* available from all the {@link Tuple}s of {@link DamageModifier} to
* {@link Function} in {@link #originalFunctions()}.
*
* @return The final damage to deal
*/
@PropertySettings(requiredParameter = false, generateMethods = false)
double finalOutputDamage();
/**
* Checks whether the provided {@link DamageModifier} is applicable to the
* current available {@link DamageModifier}s.
*
* @param damageModifier The damage modifier to check
* @return True if the damage modifier is relevant to this event
*/
boolean isModifierApplicable(DamageModifier damageModifier);
/**
* Gets the damage for the provided {@link DamageModifier}. Providing that
* {@link #isModifierApplicable(DamageModifier)} returns true
,
* the cached "damage" for the {@link DamageModifier} is returned.
*
* @param damageModifier The damage modifier to get the damage for
* @return The modifier
*/
Tuple outputDamage(DamageModifier damageModifier);
/**
* Sets the provided {@link Function} to be used for the given
* {@link DamageModifier}. If the {@link DamageModifier} is already
* included in {@link #modifiers()}, the {@link Function} replaces
* the existing function. If there is no {@link Tuple} for the
* {@link DamageModifier}, a new one is created and added to the end
* of the list of {@link Function}s to be applied to the
* {@link #baseOutputDamage()}.
*
* If needing to create a custom {@link DamageModifier} is required,
* usage of the
* {@link org.spongepowered.api.event.cause.entity.damage.DamageModifier.Builder}
* is recommended.
*
* @param damageModifier The damage modifier
* @param function The function to map to the modifier
*/
void setOutputDamage(DamageModifier damageModifier, DoubleUnaryOperator function);
/**
* Adds the provided {@link DamageModifier} and {@link Function} to the
* list of modifiers, such that the {@link Set} containing
* {@link DamageModifierType}s provided in {@code before} will appear
* after the provided damage modifier.
*
* @param damageModifier The damage modifier to add
* @param function The associated function
* @param before The set containing the modifier types to come after
* the provided modifier
*/
void addDamageModifierBefore(DamageModifier damageModifier, DoubleUnaryOperator function, Set before);
/**
* Adds the provided {@link DamageModifier} and {@link Function} to the list
* of modifiers, such that the modifier will appear in order after any
* current modifiers whose type are included in the provided {@link Set}
* of {@link DamageModifierType}s.
*
* @param damageModifier The damage modifier to add
* @param function The associated function
* @param after The set of modifier types to come before the new modifier
*/
void addDamageModifierAfter(DamageModifier damageModifier, DoubleUnaryOperator function, Set after);
/**
* Gets a list of simple {@link Tuple}s of {@link DamageModifier} keyed to
* their representative {@link Function}s. All {@link DamageModifier}s are
* applicable to the entity based on the {@link DamageSource} and any
* possible invulnerabilities due to the {@link DamageSource}.
*
* @return A list of damage modifiers to functions
*/
@PropertySettings(requiredParameter = false, generateMethods = false)
List modifiers();
/**
* Gets the knock back modifier. The modifier itself will apply to the
* momentum of the attacked entity.
*
* @return The knock back modifier
*/
float knockbackModifier();
/**
* Sets the knock back modifier. The modifier itself will apply to the
* momentum of the attacked entity.
*
* @param modifier The knock back modifier to set
*/
void setKnockbackModifier(float modifier);
}