All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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); }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy