mcMMO/src/main/java/com/gmail/nossr50/skills/utilities/CombatTools.java

585 lines
21 KiB
Java
Raw Normal View History

package com.gmail.nossr50.skills.utilities;
2012-04-27 11:47:11 +02:00
import org.bukkit.Material;
import org.bukkit.enchantments.Enchantment;
2012-04-27 11:47:11 +02:00
import org.bukkit.entity.Animals;
import org.bukkit.entity.Arrow;
2012-04-27 11:47:11 +02:00
import org.bukkit.entity.Entity;
import org.bukkit.entity.EntityType;
import org.bukkit.entity.IronGolem;
import org.bukkit.entity.LightningStrike;
2012-04-27 11:47:11 +02:00
import org.bukkit.entity.LivingEntity;
import org.bukkit.entity.Player;
import org.bukkit.entity.Skeleton;
import org.bukkit.entity.Tameable;
2012-04-27 11:47:11 +02:00
import org.bukkit.entity.Wolf;
import org.bukkit.event.entity.EntityDamageByEntityEvent;
import org.bukkit.event.entity.EntityDamageEvent;
import org.bukkit.event.entity.EntityDamageEvent.DamageCause;
import org.bukkit.event.player.PlayerAnimationEvent;
2012-04-27 11:47:11 +02:00
import org.bukkit.inventory.ItemStack;
import com.gmail.nossr50.mcMMO;
2012-04-27 11:47:11 +02:00
import com.gmail.nossr50.config.Config;
import com.gmail.nossr50.datatypes.PlayerProfile;
import com.gmail.nossr50.events.fake.FakeEntityDamageByEntityEvent;
import com.gmail.nossr50.events.fake.FakeEntityDamageEvent;
import com.gmail.nossr50.locale.LocaleLoader;
2012-06-08 23:48:41 +02:00
import com.gmail.nossr50.party.PartyManager;
import com.gmail.nossr50.skills.acrobatics.Acrobatics;
import com.gmail.nossr50.skills.acrobatics.AcrobaticsManager;
import com.gmail.nossr50.skills.archery.Archery;
2012-06-13 03:36:01 +02:00
import com.gmail.nossr50.skills.archery.ArcheryManager;
2013-01-11 04:39:08 +01:00
import com.gmail.nossr50.skills.axes.AxeManager;
import com.gmail.nossr50.skills.axes.Axes;
import com.gmail.nossr50.skills.runnables.BleedTimer;
import com.gmail.nossr50.skills.runnables.GainXp;
2012-06-18 14:50:14 +02:00
import com.gmail.nossr50.skills.swords.Swords;
2012-06-21 15:04:45 +02:00
import com.gmail.nossr50.skills.swords.SwordsManager;
import com.gmail.nossr50.skills.taming.Taming;
2012-06-12 17:07:51 +02:00
import com.gmail.nossr50.skills.taming.TamingManager;
import com.gmail.nossr50.skills.unarmed.Unarmed;
2012-06-15 03:10:47 +02:00
import com.gmail.nossr50.skills.unarmed.UnarmedManager;
import com.gmail.nossr50.util.ItemChecks;
import com.gmail.nossr50.util.Misc;
import com.gmail.nossr50.util.Permissions;
import com.gmail.nossr50.util.Users;
2012-04-27 11:47:11 +02:00
public final class CombatTools {
private static Config configInstance = Config.getInstance();
2012-04-27 11:47:11 +02:00
private CombatTools() {}
2013-01-26 23:01:55 +01:00
2012-04-27 11:47:11 +02:00
/**
* Apply combat modifiers and process and XP gain.
*
* @param event The event to run the combat checks on.
*/
public static void combatChecks(EntityDamageByEntityEvent event, Entity attacker, LivingEntity target) {
2012-06-22 18:54:49 +02:00
boolean targetIsPlayer = (target.getType() == EntityType.PLAYER);
boolean targetIsTamedPet = (target instanceof Tameable) ? ((Tameable) target).isTamed() : false;
if (attacker instanceof Player) {
Player player = (Player) attacker;
2012-04-27 11:47:11 +02:00
if (Misc.isNPCPlayer(player)) {
return;
}
ItemStack heldItem = player.getItemInHand();
Material heldItemType = heldItem.getType();
if (ItemChecks.isSword(heldItem)) {
2012-06-22 18:54:49 +02:00
if (targetIsPlayer || targetIsTamedPet) {
if (!Swords.pvpEnabled) {
return;
}
}
else if (!Swords.pveEnabled) {
2012-06-22 18:54:49 +02:00
return;
}
if (Permissions.swords(player)) {
SwordsManager swordsManager = new SwordsManager(player);
PlayerProfile profile = swordsManager.getProfile();
boolean canSerratedStrike = Permissions.serratedStrikes(player); //So we don't have to check the same permission twice
if (profile.getToolPreparationMode(ToolType.SWORD) && canSerratedStrike) {
2013-01-26 23:01:55 +01:00
SkillTools.abilityCheck(player, SkillType.SWORDS);
}
2012-06-22 20:20:28 +02:00
if (Permissions.swordsBleed(player) && shouldBeAffected(player, target)) {
swordsManager.bleedCheck(target);
}
2012-04-27 11:47:11 +02:00
if (profile.getAbilityMode(AbilityType.SERRATED_STRIKES) && canSerratedStrike) {
swordsManager.serratedStrikes(target, event.getDamage());
}
startGainXp(player, profile, target, SkillType.SWORDS);
}
2012-04-27 11:47:11 +02:00
}
else if (ItemChecks.isAxe(heldItem)) {
2012-06-22 18:54:49 +02:00
if (targetIsPlayer || targetIsTamedPet) {
if (!Axes.pvpEnabled) {
return;
}
}
else if (!Axes.pveEnabled) {
2012-06-22 18:54:49 +02:00
return;
}
if (Permissions.axes(player)) {
AxeManager axeManager = new AxeManager(player);
PlayerProfile profile = axeManager.getProfile();
boolean canSkullSplit = Permissions.skullSplitter(player); //So we don't have to check the same permission twice
if (profile.getToolPreparationMode(ToolType.AXE) && canSkullSplit) {
2013-01-26 23:01:55 +01:00
SkillTools.abilityCheck(player, SkillType.AXES);
}
if (Permissions.axeBonus(player)) {
axeManager.bonusDamage(event);
}
2012-06-22 20:20:28 +02:00
if (!target.isDead() && Permissions.criticalHit(player) && shouldBeAffected(player, target)) {
axeManager.criticalHitCheck(event, target);
}
if (!target.isDead() && Permissions.impact(player)) {
axeManager.impact(event, target);
}
2012-06-05 15:57:10 +02:00
if (!target.isDead() && profile.getAbilityMode(AbilityType.SKULL_SPLIITER) && canSkullSplit) {
axeManager.skullSplitter(target, event.getDamage());
}
startGainXp(player, profile, target, SkillType.AXES);
}
2012-04-27 11:47:11 +02:00
}
else if (heldItemType == Material.AIR) {
2012-06-22 18:54:49 +02:00
if (targetIsPlayer || targetIsTamedPet) {
if (!configInstance.getUnarmedPVP()) {
return;
}
}
2012-06-22 18:54:49 +02:00
else if (!configInstance.getUnarmedPVE()) {
return;
}
if (Permissions.unarmed(player)) {
UnarmedManager unarmedManager = new UnarmedManager(player);
PlayerProfile profile = unarmedManager.getProfile();
boolean canBerserk = Permissions.berserk(player); //So we don't have to check the same permission twice
if (profile.getToolPreparationMode(ToolType.FISTS) && canBerserk) {
2013-01-26 23:01:55 +01:00
SkillTools.abilityCheck(player, SkillType.UNARMED);
}
if (Permissions.unarmedBonus(player)) {
unarmedManager.bonusDamage(event);
}
if (profile.getAbilityMode(AbilityType.BERSERK) && canBerserk) {
unarmedManager.berserkDamage(event);
}
2012-06-22 20:20:28 +02:00
if (target instanceof Player && Permissions.disarm(player)) {
unarmedManager.disarmCheck(target);
}
2012-04-27 11:47:11 +02:00
startGainXp(player, unarmedManager.getProfile(), target, SkillType.UNARMED);
}
2012-04-27 11:47:11 +02:00
}
else if (heldItemType == Material.BONE && target instanceof Tameable && Permissions.beastLore(player)) {
TamingManager tamingManager = new TamingManager(player);
2012-06-12 20:42:38 +02:00
tamingManager.beastLore(target);
event.setCancelled(true);
2012-04-27 11:47:11 +02:00
}
}
2012-06-22 18:54:49 +02:00
Entity damager = event.getDamager();
2012-07-03 03:05:55 +02:00
switch (damager.getType()) {
2012-06-22 18:54:49 +02:00
case WOLF:
Wolf wolf = (Wolf) damager;
2012-04-27 11:47:11 +02:00
if (wolf.isTamed() && wolf.getOwner() instanceof Player) {
Player master = (Player) wolf.getOwner();
if (Misc.isNPCPlayer(master)) {
return;
}
2012-06-22 18:54:49 +02:00
if (targetIsPlayer || targetIsTamedPet) {
if (!Taming.pvpEnabled) {
return;
}
}
else if (!Taming.pveEnabled) {
2012-06-22 18:54:49 +02:00
return;
}
if (Permissions.taming(master)) {
TamingManager tamingManager = new TamingManager(master);
int skillLevel = tamingManager.getSkillLevel();
if (skillLevel >= Taming.fastFoodServiceUnlockLevel && Permissions.fastFoodService(master)) {
tamingManager.fastFoodService(wolf, event.getDamage());
}
if (skillLevel >= Taming.sharpenedClawsUnlockLevel && Permissions.sharpenedClaws(master)) {
tamingManager.sharpenedClaws(event);
}
2012-04-27 11:47:11 +02:00
if (Permissions.gore(master)) {
tamingManager.gore(event);
}
if (target != master) {
startGainXp(master, tamingManager.getProfile(), target, SkillType.TAMING);
}
}
2012-04-27 11:47:11 +02:00
}
2012-06-22 18:54:49 +02:00
break;
2012-07-03 03:05:55 +02:00
2012-06-22 18:54:49 +02:00
case ARROW:
LivingEntity shooter = ((Arrow) damager).getShooter();
2012-06-05 16:31:55 +02:00
/* Break instead of return due to Dodge/Counter/Deflect abilities */
if (shooter == null || !(shooter instanceof Player)) {
2012-06-22 19:39:14 +02:00
break;
}
if (targetIsPlayer || targetIsTamedPet) {
if (!Archery.pvpEnabled) {
return;
}
}
else if (!Archery.pveEnabled) {
return;
}
2012-06-05 16:31:55 +02:00
archeryCheck((Player) shooter, target, event);
2012-06-22 18:54:49 +02:00
break;
default:
break;
2012-04-27 11:47:11 +02:00
}
2012-06-22 19:39:14 +02:00
if (targetIsPlayer) {
2012-06-21 15:04:45 +02:00
Player player = (Player) target;
if (Misc.isNPCPlayer(player)) {
return;
}
ItemStack heldItem = player.getItemInHand();
if (damager instanceof Player) {
if (Swords.pvpEnabled && ItemChecks.isSword(heldItem) && Permissions.counterAttack(player)) {
SwordsManager swordsManager = new SwordsManager(player);
swordsManager.counterAttackChecks((LivingEntity) damager, event.getDamage());
}
if (Acrobatics.pvpEnabled && Permissions.dodge(player)) {
AcrobaticsManager acrobaticsManager = new AcrobaticsManager(player);
acrobaticsManager.dodgeCheck(event);
2012-06-22 18:54:49 +02:00
}
if (Unarmed.pvpEnabled && heldItem.getType() == Material.AIR && Permissions.deflect(player)) {
UnarmedManager unarmedManager = new UnarmedManager(player);
unarmedManager.deflectCheck(event);
}
}
2012-06-22 18:54:49 +02:00
else {
if (Swords.pveEnabled && damager instanceof LivingEntity && ItemChecks.isSword(heldItem) && Permissions.counterAttack(player)) {
SwordsManager swordsManager = new SwordsManager(player);
swordsManager.counterAttackChecks((LivingEntity) damager, event.getDamage());
2012-06-22 18:54:49 +02:00
}
if (Acrobatics.pveEnabled && !(damager instanceof LightningStrike && Acrobatics.dodgeLightningDisabled) && Permissions.dodge(player)) {
AcrobaticsManager acrobaticsManager = new AcrobaticsManager(player);
acrobaticsManager.dodgeCheck(event);
2012-06-22 18:54:49 +02:00
}
}
2012-04-27 11:47:11 +02:00
}
}
/**
* Process archery abilities.
*
2012-06-05 16:31:55 +02:00
* @param shooter The player shooting
* @param target The defending entity
2012-04-27 11:47:11 +02:00
* @param event The event to run the archery checks on.
*/
private static void archeryCheck(Player shooter, LivingEntity target, EntityDamageByEntityEvent event) {
if (Misc.isNPCPlayer(shooter)) {
return;
2012-04-27 11:47:11 +02:00
}
if (Permissions.archery(shooter)) {
ArcheryManager archeryManager = new ArcheryManager(shooter);
archeryManager.skillShot(event);
if (target instanceof Player && Permissions.daze(shooter)) {
archeryManager.dazeCheck((Player) target, event);
}
2012-04-27 11:47:11 +02:00
if (!(shooter.getItemInHand().containsEnchantment(Enchantment.ARROW_INFINITE)) && Permissions.trackArrows(shooter)) {
archeryManager.trackArrows(target);
}
2012-06-13 03:36:01 +02:00
if (target != shooter) {
archeryManager.distanceXpBonus(target);
startGainXp(shooter, archeryManager.getProfile(), target, SkillType.ARCHERY);
}
2012-04-27 11:47:11 +02:00
}
}
/**
* Attempt to damage target for value dmg with reason CUSTOM
*
* @param target LivingEntity which to attempt to damage
* @param dmg Amount of damage to attempt to do
*/
public static void dealDamage(LivingEntity target, int dmg) {
dealDamage(target, dmg, EntityDamageEvent.DamageCause.CUSTOM);
}
/**
* Attempt to damage target for value dmg with reason cause
*
* @param target LivingEntity which to attempt to damage
* @param dmg Amount of damage to attempt to do
* @param cause DamageCause to pass to damage event
*/
private static void dealDamage(LivingEntity target, int dmg, DamageCause cause) {
if (configInstance.getEventCallbackEnabled()) {
2012-06-05 15:57:10 +02:00
EntityDamageEvent ede = new FakeEntityDamageEvent(target, cause, dmg);
mcMMO.p.getServer().getPluginManager().callEvent(ede);
2012-04-27 11:47:11 +02:00
if (ede.isCancelled()) {
return;
}
target.damage(ede.getDamage());
}
else {
target.damage(dmg);
}
}
/**
* Attempt to damage target for value dmg with reason ENTITY_ATTACK with damager attacker
*
* @param target LivingEntity which to attempt to damage
* @param dmg Amount of damage to attempt to do
* @param attacker Player to pass to event as damager
*/
private static void dealDamage(LivingEntity target, int dmg, Player attacker) {
if (configInstance.getEventCallbackEnabled()) {
2012-06-05 15:57:10 +02:00
EntityDamageEvent ede = new FakeEntityDamageByEntityEvent(attacker, target, EntityDamageEvent.DamageCause.ENTITY_ATTACK, dmg);
mcMMO.p.getServer().getPluginManager().callEvent(ede);
2012-04-27 11:47:11 +02:00
if (ede.isCancelled()) {
return;
}
target.damage(ede.getDamage());
}
else {
target.damage(dmg);
}
}
/**
* Apply Area-of-Effect ability actions.
*
* @param attacker The attacking player
* @param target The defending entity
* @param damage The initial damage amount
* @param type The type of skill being used
*/
2012-06-21 15:04:45 +02:00
public static void applyAbilityAoE(Player attacker, LivingEntity target, int damage, SkillType type) {
2012-04-27 11:47:11 +02:00
int numberOfTargets = Misc.getTier(attacker.getItemInHand()); //The higher the weapon tier, the more targets you hit
int damageAmount = damage;
2012-04-27 11:47:11 +02:00
if (damageAmount < 1) {
damageAmount = 1;
}
for (Entity entity : target.getNearbyEntities(2.5, 2.5, 2.5)) {
if ((entity instanceof Player && Misc.isNPCPlayer((Player) entity)) || !(entity instanceof LivingEntity) || !shouldBeAffected(attacker, entity)) {
continue;
}
2012-05-01 01:32:50 +02:00
if (numberOfTargets <= 0) {
break;
}
2012-05-01 01:32:50 +02:00
2012-06-18 14:50:14 +02:00
PlayerAnimationEvent armswing = new PlayerAnimationEvent(attacker);
mcMMO.p.getServer().getPluginManager().callEvent(armswing);
switch (type) {
case SWORDS:
if (entity instanceof Player) {
((Player) entity).sendMessage(LocaleLoader.getString("Swords.Combat.SS.Struck"));
2012-04-27 11:47:11 +02:00
}
2012-05-01 01:32:50 +02:00
2013-01-22 16:48:10 +01:00
BleedTimer.add((LivingEntity) entity, Swords.serratedStrikesBleedTicks);
2012-05-01 01:32:50 +02:00
break;
case AXES:
if (entity instanceof Player) {
((Player) entity).sendMessage(LocaleLoader.getString("Axes.Combat.Cleave.Struck"));
2012-04-27 11:47:11 +02:00
}
2012-05-01 01:32:50 +02:00
break;
default:
break;
2012-04-27 11:47:11 +02:00
}
2012-05-01 01:32:50 +02:00
dealDamage((LivingEntity) entity, damageAmount, attacker);
numberOfTargets--;
2012-04-27 11:47:11 +02:00
}
}
/**
* Start the task that gives combat XP.
*
* @param attacker The attacking player
2012-07-03 16:04:04 +02:00
* @param profile The player's PlayerProfile
2012-04-27 11:47:11 +02:00
* @param target The defending entity
* @param skillType The skill being used
*/
2012-07-03 16:04:04 +02:00
public static void startGainXp(Player attacker, PlayerProfile profile, LivingEntity target, SkillType skillType) {
2012-04-27 11:47:11 +02:00
double baseXP = 0;
if (target instanceof Player) {
if (!configInstance.getExperienceGainsPlayerVersusPlayerEnabled()) {
2012-04-27 11:47:11 +02:00
return;
}
Player defender = (Player) target;
2012-06-12 05:32:56 +02:00
if (System.currentTimeMillis() >= Users.getProfile(defender).getRespawnATS() + 5) {
baseXP = 20 * configInstance.getPlayerVersusPlayerXP();
2012-04-27 11:47:11 +02:00
}
}
2013-01-07 23:08:53 +01:00
else if (!mcMMO.placeStore.isSpawnedMob(target)) {
if (target instanceof Animals && !mcMMO.placeStore.isSpawnedPet(target)) {
baseXP = configInstance.getAnimalsXP();
2012-04-27 11:47:11 +02:00
}
else {
EntityType type = target.getType();
switch (type) {
case BAT:
baseXP = configInstance.getAnimalsXP();
break;
2012-04-27 11:47:11 +02:00
case BLAZE:
baseXP = configInstance.getBlazeXP();
2012-04-27 11:47:11 +02:00
break;
case CAVE_SPIDER:
baseXP = configInstance.getCaveSpiderXP();
2012-04-27 11:47:11 +02:00
break;
case CREEPER:
baseXP = configInstance.getCreeperXP();
2012-04-27 11:47:11 +02:00
break;
case ENDER_DRAGON:
baseXP = configInstance.getEnderDragonXP();
2012-04-27 11:47:11 +02:00
break;
case ENDERMAN:
baseXP = configInstance.getEndermanXP();
2012-04-27 11:47:11 +02:00
break;
case GHAST:
baseXP = configInstance.getGhastXP();
2012-04-27 11:47:11 +02:00
break;
case GIANT:
baseXP = configInstance.getGiantXP();
break;
2012-04-27 11:47:11 +02:00
case MAGMA_CUBE:
baseXP = configInstance.getMagmaCubeXP();
2012-04-27 11:47:11 +02:00
break;
case IRON_GOLEM:
2012-05-24 16:35:46 +02:00
if (!((IronGolem) target).isPlayerCreated()) {
baseXP = configInstance.getIronGolemXP();
2012-05-24 16:35:46 +02:00
}
2012-04-27 11:47:11 +02:00
break;
case PIG_ZOMBIE:
baseXP = configInstance.getPigZombieXP();
2012-04-27 11:47:11 +02:00
break;
case SILVERFISH:
baseXP = configInstance.getSilverfishXP();
2012-04-27 11:47:11 +02:00
break;
case SKELETON:
switch(((Skeleton) target).getSkeletonType()) {
case WITHER:
baseXP = configInstance.getWitherSkeletonXP();
break;
default:
baseXP = configInstance.getSkeletonXP();
break;
}
2012-04-27 11:47:11 +02:00
case SLIME:
baseXP = configInstance.getSlimeXP();
2012-04-27 11:47:11 +02:00
break;
case SPIDER:
baseXP = configInstance.getSpiderXP();
2012-04-27 11:47:11 +02:00
break;
case WITCH:
baseXP = configInstance.getWitchXP();
break;
case WITHER:
baseXP = configInstance.getWitherXP();
break;
2012-04-27 11:47:11 +02:00
case ZOMBIE:
baseXP = configInstance.getZombieXP();
2012-04-27 11:47:11 +02:00
break;
default:
break;
}
}
baseXP *= 10;
}
if (baseXP != 0) {
2012-07-03 16:04:04 +02:00
mcMMO.p.getServer().getScheduler().scheduleSyncDelayedTask(mcMMO.p, new GainXp(attacker, profile, skillType, baseXP, target), 0);
2012-04-27 11:47:11 +02:00
}
}
2012-06-18 14:50:14 +02:00
/**
* Check to see if the given LivingEntity should be affected by a combat ability.
*
* @param player The attacking Player
* @param livingEntity The defending LivingEntity
* @return true if the Entity should be damaged, false otherwise.
2012-06-18 14:50:14 +02:00
*/
public static boolean shouldBeAffected(Player player, Entity entity) {
if (entity instanceof Player) {
Player defender = (Player) entity;
2012-06-21 15:04:45 +02:00
2013-01-26 23:01:55 +01:00
if (!defender.getWorld().getPVP() || defender == player || PartyManager.inSameParty(player, defender) || Users.getProfile(defender).getGodMode()) {
2012-06-21 15:04:45 +02:00
return false;
}
2012-12-24 22:56:25 +01:00
//It may seem a bit redundant but we need a check here to prevent bleed from being applied in applyAbilityAoE()
EntityDamageEvent ede = new FakeEntityDamageByEntityEvent(player, entity, EntityDamageEvent.DamageCause.ENTITY_ATTACK, 1);
mcMMO.p.getServer().getPluginManager().callEvent(ede);
if (ede.isCancelled()) {
return false;
}
}
else if (entity instanceof Tameable) {
if (Misc.isFriendlyPet(player, (Tameable) entity)) {
return false;
2012-06-18 14:50:14 +02:00
}
}
2012-06-21 15:04:45 +02:00
return true;
2012-06-18 14:50:14 +02:00
}
2012-04-27 11:47:11 +02:00
}