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

617 lines
23 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;
import org.bukkit.entity.AnimalTamer;
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.McMMOPlayer;
2012-04-27 11:47:11 +02:00
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;
import com.gmail.nossr50.mods.ModChecks;
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;
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.runnables.BleedTimer;
import com.gmail.nossr50.skills.runnables.CombatXpGiver;
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;
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.isNPCEntity(player)) {
return;
}
if (target instanceof Tameable && isFriendlyPet(player, (Tameable) target)) {
return;
}
ItemStack heldItem = player.getItemInHand();
Material heldItemType = heldItem.getType();
DamageCause damageCause = event.getCause();
if (ItemChecks.isSword(heldItem) && damageCause == DamageCause.ENTITY_ATTACK) {
2012-06-22 18:54:49 +02:00
if (targetIsPlayer || targetIsTamedPet) {
if (!SkillType.SWORDS.getPVPEnabled()) {
return;
}
}
else if (!SkillType.SWORDS.getPVEEnabled()) {
2012-06-22 18:54:49 +02:00
return;
}
if (Permissions.skillEnabled(player, SkillType.SWORDS)) {
McMMOPlayer mcMMOPlayer = Users.getPlayer(player);
PlayerProfile profile = mcMMOPlayer.getProfile();
SwordsManager swordsManager = new SwordsManager(mcMMOPlayer);
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.bleed(player)) {
swordsManager.bleedCheck(target);
}
2012-04-27 11:47:11 +02:00
if (profile.getAbilityMode(AbilityType.SERRATED_STRIKES) && canSerratedStrike) {
swordsManager.serratedStrikes(target, event.getDamage());
}
startGainXp(mcMMOPlayer, target, SkillType.SWORDS);
}
2012-04-27 11:47:11 +02:00
}
else if (ItemChecks.isAxe(heldItem) && damageCause == DamageCause.ENTITY_ATTACK) {
2012-06-22 18:54:49 +02:00
if (targetIsPlayer || targetIsTamedPet) {
if (!SkillType.AXES.getPVPEnabled()) {
return;
}
}
else if (!SkillType.AXES.getPVEEnabled()) {
2012-06-22 18:54:49 +02:00
return;
}
if (Permissions.skillEnabled(player, SkillType.AXES)) {
McMMOPlayer mcMMOPlayer = Users.getPlayer(player);
PlayerProfile profile = mcMMOPlayer.getProfile();
AxeManager axeManager = new AxeManager(mcMMOPlayer);
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.bonusDamage(player, axeManager.getSkill())) {
axeManager.bonusDamage(event);
}
2012-06-22 20:20:28 +02:00
if (!target.isDead() && Permissions.criticalStrikes(player)) {
axeManager.criticalHitCheck(event, target);
}
if (!target.isDead() && Permissions.armorImpact(player)) {
axeManager.impact(event, target);
}
2012-06-05 15:57:10 +02:00
if (!target.isDead() && profile.getAbilityMode(AbilityType.SKULL_SPLITTER) && canSkullSplit) {
axeManager.skullSplitter(target, event.getDamage());
}
startGainXp(mcMMOPlayer, target, SkillType.AXES);
}
2012-04-27 11:47:11 +02:00
}
else if (heldItemType == Material.AIR && damageCause == DamageCause.ENTITY_ATTACK) {
2012-06-22 18:54:49 +02:00
if (targetIsPlayer || targetIsTamedPet) {
if (!SkillType.UNARMED.getPVPEnabled()) {
return;
}
}
else if (!SkillType.UNARMED.getPVEEnabled()) {
2012-06-22 18:54:49 +02:00
return;
}
if (Permissions.skillEnabled(player, SkillType.UNARMED)) {
McMMOPlayer mcMMOPlayer = Users.getPlayer(player);
PlayerProfile profile = mcMMOPlayer.getProfile();
UnarmedManager unarmedManager = new UnarmedManager(mcMMOPlayer);
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.bonusDamage(player, unarmedManager.getSkill())) {
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(mcMMOPlayer, target, SkillType.UNARMED);
}
2012-04-27 11:47:11 +02:00
}
else if (heldItemType == Material.BONE && target instanceof Tameable && Permissions.beastLore(player) && damageCause == DamageCause.ENTITY_ATTACK) {
TamingManager tamingManager = new TamingManager(Users.getPlayer(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.isNPCEntity(master)) {
return;
}
2012-06-22 18:54:49 +02:00
if (targetIsPlayer || targetIsTamedPet) {
if (!SkillType.TAMING.getPVPEnabled()) {
return;
}
}
else if (!SkillType.TAMING.getPVEEnabled()) {
2012-06-22 18:54:49 +02:00
return;
}
if (Permissions.skillEnabled(master, SkillType.TAMING)) {
McMMOPlayer mcMMOPlayer = Users.getPlayer(master);
TamingManager tamingManager = new TamingManager(mcMMOPlayer);
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);
}
startGainXp(mcMMOPlayer, 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 (!SkillType.ARCHERY.getPVPEnabled()) {
return;
}
}
else if (!SkillType.ARCHERY.getPVEEnabled()) {
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.isNPCEntity(player)) {
return;
}
ItemStack heldItem = player.getItemInHand();
if (damager instanceof Player) {
if (SkillType.SWORDS.getPVPEnabled() && ItemChecks.isSword(heldItem) && Permissions.counterAttack(player)) {
SwordsManager swordsManager = new SwordsManager(Users.getPlayer(player));
swordsManager.counterAttackChecks((LivingEntity) damager, event.getDamage());
}
if (SkillType.ACROBATICS.getPVPEnabled() && Permissions.dodge(player)) {
AcrobaticsManager acrobaticsManager = new AcrobaticsManager(Users.getPlayer(player));
acrobaticsManager.dodgeCheck(event);
2012-06-22 18:54:49 +02:00
}
}
2012-06-22 18:54:49 +02:00
else {
if (SkillType.SWORDS.getPVEEnabled() && damager instanceof LivingEntity && ItemChecks.isSword(heldItem) && Permissions.counterAttack(player)) {
SwordsManager swordsManager = new SwordsManager(Users.getPlayer(player));
swordsManager.counterAttackChecks((LivingEntity) damager, event.getDamage());
2012-06-22 18:54:49 +02:00
}
if (SkillType.ACROBATICS.getPVEEnabled() && !(damager instanceof LightningStrike && Acrobatics.dodgeLightningDisabled) && Permissions.dodge(player)) {
AcrobaticsManager acrobaticsManager = new AcrobaticsManager(Users.getPlayer(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.isNPCEntity(shooter)) {
return;
2012-04-27 11:47:11 +02:00
}
if (Permissions.skillEnabled(shooter, SkillType.ARCHERY)) {
McMMOPlayer mcMMOPlayer = Users.getPlayer(shooter);
ArcheryManager archeryManager = new ArcheryManager(mcMMOPlayer);
archeryManager.skillShot(event);
2013-02-02 20:23:40 +01:00
if (target instanceof Player) {
if (SkillType.UNARMED.getPVPEnabled() && ((Player) target).getItemInHand().getType() == Material.AIR && Permissions.arrowDeflect((Player) target)) {
2013-02-02 20:23:40 +01:00
UnarmedManager unarmedManager = new UnarmedManager(Users.getPlayer((Player) target));
unarmedManager.deflectCheck(event);
}
if (Permissions.daze(shooter)) {
archeryManager.dazeCheck((Player) target, event);
}
}
2012-04-27 11:47:11 +02:00
if (!(shooter.getItemInHand().containsEnchantment(Enchantment.ARROW_INFINITE)) && Permissions.arrowRetrieval(shooter)) {
archeryManager.trackArrows(target);
}
2012-06-13 03:36:01 +02:00
archeryManager.distanceXpBonus(target);
startGainXp(mcMMOPlayer, 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 (Misc.isNPCEntity(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 mcMMOPlayer The attacking player
2012-04-27 11:47:11 +02:00
* @param target The defending entity
* @param skillType The skill being used
*/
public static void startGainXp(McMMOPlayer mcMMOPlayer, 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;
if (System.currentTimeMillis() >= Users.getPlayer(defender).getProfile().getRespawnATS() + 5) {
baseXP = 20 * configInstance.getPlayerVersusPlayerXP();
2012-04-27 11:47:11 +02:00
}
}
else if (!target.hasMetadata(mcMMO.entityMetadataKey)) {
if (target instanceof Animals) {
if (ModChecks.isCustomEntity(target)) {
baseXP = ModChecks.getCustomEntity(target).getXpMultiplier();
}
else {
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:
case CAVE_SPIDER:
case CREEPER:
case ENDER_DRAGON:
case ENDERMAN:
case GHAST:
case GIANT:
2012-04-27 11:47:11 +02:00
case MAGMA_CUBE:
case PIG_ZOMBIE:
case SILVERFISH:
case SLIME:
case SPIDER:
case WITCH:
case WITHER:
case ZOMBIE:
baseXP = Config.getInstance().getCombatXP(type);
2012-04-27 11:47:11 +02:00
break;
// Temporary workaround for custom entities
case UNKNOWN:
baseXP = 1.0;
2012-04-27 11:47:11 +02:00
break;
case SKELETON:
switch(((Skeleton) target).getSkeletonType()) {
case WITHER:
baseXP = configInstance.getWitherSkeletonXP();
break;
default:
baseXP = Config.getInstance().getCombatXP(type);
break;
}
break;
case IRON_GOLEM:
if (!((IronGolem) target).isPlayerCreated()) {
baseXP = Config.getInstance().getCombatXP(type);
}
break;
2012-04-27 11:47:11 +02:00
default:
if (ModChecks.isCustomEntity(target)) {
baseXP = ModChecks.getCustomEntity(target).getXpMultiplier();
}
2012-04-27 11:47:11 +02:00
break;
}
}
baseXP *= 10;
}
if (baseXP != 0) {
mcMMO.p.getServer().getScheduler().scheduleSyncDelayedTask(mcMMO.p, new CombatXpGiver(mcMMOPlayer, 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 entity The defending Entity
* @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
if (!defender.getWorld().getPVP() || defender == player || Users.getPlayer(defender).getProfile().getGodMode()) {
return false;
}
if (PartyManager.inSameParty(player, defender) && !(Permissions.friendlyFire(player) && Permissions.friendlyFire(defender))) {
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 (isFriendlyPet(player, (Tameable) entity)) {
// isFriendlyPet ensures that the Tameable is: Tamed, owned by a player, and the owner is in the same party
// So we can make some assumptions here, about our casting and our check
Player owner = (Player) ((Tameable) entity).getOwner();
if (!(Permissions.friendlyFire(player) && Permissions.friendlyFire(owner))) {
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
}
/**
* Checks to see if an entity is currently invincible.
*
* @param le The LivingEntity to check
* @param event The event the entity is involved in
* @return true if the entity is invincible, false otherwise
*/
public static boolean isInvincible(LivingEntity le, EntityDamageEvent event) {
/*
* So apparently if you do more damage to a LivingEntity than its last damage int you bypass the invincibility.
* So yeah, this is for that.
*/
if (le.getNoDamageTicks() > le.getMaximumNoDamageTicks() / 2.0F && event.getDamage() <= le.getLastDamage()) {
return true;
}
return false;
}
/**
* Checks to see if an entity is currently friendly toward a given player.
*
* @param attacker The player to check.
* @param pet The entity to check.
* @return true if the entity is friendly, false otherwise
*/
public static boolean isFriendlyPet(Player attacker, Tameable pet) {
if (pet.isTamed()) {
AnimalTamer tamer = pet.getOwner();
if (tamer instanceof Player) {
Player owner = (Player) tamer;
if (owner == attacker || PartyManager.inSameParty(attacker, owner)) {
return true;
}
}
}
return false;
}
2012-04-27 11:47:11 +02:00
}