/*******************************************************************************
* Copyright 2014 Tob
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
******************************************************************************/
package de.tobiyas.racesandclasses.traitcontainer.traits.defaultraits.magic.ExplosionTrait;
import java.util.LinkedList;
import java.util.List;
import java.util.UUID;
import org.bukkit.ChatColor;
import org.bukkit.Location;
import org.bukkit.block.BlockFace;
import org.bukkit.entity.Entity;
import org.bukkit.entity.LivingEntity;
import org.bukkit.event.EventHandler;
import org.bukkit.event.entity.EntityDamageByEntityEvent;
import org.bukkit.event.entity.EntityDamageEvent;
import org.bukkit.event.entity.EntityDamageEvent.DamageCause;
import de.tobiyas.racesandclasses.playermanagement.player.RaCPlayer;
import de.tobiyas.racesandclasses.traitcontainer.interfaces.TraitResults;
import de.tobiyas.racesandclasses.traitcontainer.interfaces.annotations.configuration.TraitConfigurationField;
import de.tobiyas.racesandclasses.traitcontainer.interfaces.annotations.configuration.TraitConfigurationNeeded;
import de.tobiyas.racesandclasses.traitcontainer.interfaces.annotations.configuration.TraitEventsUsed;
import de.tobiyas.racesandclasses.traitcontainer.interfaces.annotations.configuration.TraitInfos;
import de.tobiyas.racesandclasses.traitcontainer.interfaces.markerinterfaces.CostType;
import de.tobiyas.racesandclasses.traitcontainer.interfaces.markerinterfaces.Trait;
import de.tobiyas.racesandclasses.traitcontainer.traits.magic.AbstractMagicSpellTrait;
import de.tobiyas.racesandclasses.util.bukkit.versioning.compatibility.CompatibilityModifier;
import de.tobiyas.racesandclasses.util.traitutil.TraitConfiguration;
import de.tobiyas.racesandclasses.util.traitutil.TraitConfigurationFailedException;
public class ExplosionTrait extends AbstractMagicSpellTrait {
/**
* Tells to deal world damage if true
*/
private boolean explode = false;
/**
* the range to explode
*/
private int range = 0;
/**
* The damage to deal on explosion
*/
private double damage = 0;
@TraitEventsUsed()
@Override
public void generalInit() {
}
@Override
public String getName() {
return "ExplosionTrait";
}
@Override
protected String getPrettyConfigIntern(){
return costType.name() +
(costType == CostType.ITEM ?
(" " + materialForCasting.name() + " ")
: " "
)
+ cost;
}
@TraitInfos(category="magic", traitName="ExplosionTrait", visible=true)
@Override
public void importTrait() {
}
@TraitConfigurationNeeded(fields = {
@TraitConfigurationField(fieldName = "damage", classToExpect = Double.class ),
@TraitConfigurationField(fieldName = "range", classToExpect = Integer.class ),
@TraitConfigurationField(fieldName = "explode", classToExpect = Boolean.class, optional = true)
})
@Override
public void setConfiguration(TraitConfiguration configMap) throws TraitConfigurationFailedException {
super.setConfiguration(configMap);
this.damage = configMap.getAsDouble("damage");
this.range = configMap.getAsInt("range");
if(configMap.containsKey("explode")){
this.explode = configMap.getAsBool("explode");
}
}
@Override
public boolean isBetterThan(Trait trait) {
if(!(trait instanceof ExplosionTrait)) return false;
ExplosionTrait otherTrait = (ExplosionTrait) trait;
return cost > otherTrait.cost;
}
public static List<String> getHelpForTrait(){
List<String> helpList = new LinkedList<String>();
helpList.add(ChatColor.YELLOW + "This trait lets you explode and deal damage to everyone around you.");
return helpList;
}
@Override
protected void magicSpellTriggered(RaCPlayer player, TraitResults result) {
Location location = player.getPlayer().getEyeLocation();
player.getPlayer().setNoDamageTicks(10);
location.getWorld().createExplosion(location.getBlock().getRelative(BlockFace.NORTH).getLocation(), 0);
location.getWorld().createExplosion(location.getBlock().getRelative(BlockFace.EAST).getLocation(), 0);
location.getWorld().createExplosion(location.getBlock().getRelative(BlockFace.SOUTH).getLocation(), 0);
location.getWorld().createExplosion(location.getBlock().getRelative(BlockFace.WEST).getLocation(), 0);
double modDamage = modifyToPlayer(player, damage, "damage");
if(explode){
lastCaster = player.getUniqueId();
if(location.getWorld().createExplosion(location, (float) modDamage)){
result.setTriggered(true);
lastCaster = null;
return;
}
lastCaster = null;
}
List<LivingEntity> entities = getNearbyEntities(location, range);
for(LivingEntity entity : entities){
if(entity == player){
continue;
}
EntityDamageByEntityEvent damageEvent = CompatibilityModifier.EntityDamageByEntity.
safeCreateEvent(player.getPlayer(), entity, DamageCause.ENTITY_EXPLOSION, modDamage);
lastCaster = player.getUniqueId();
plugin.fireEventToBukkit(damageEvent);
lastCaster = null;
double newDamage = CompatibilityModifier.EntityDamage.safeGetDamage(damageEvent);
if(!damageEvent.isCancelled() && newDamage > 0){
de.tobiyas.racesandclasses.util.bukkit.versioning.compatibility.CompatibilityModifier.LivingEntity
.safeDamageEntityByEntity(entity, player.getPlayer(), newDamage);
}
}
result.setTriggered(true);
}
private UUID lastCaster;
@EventHandler
public void onDamage(EntityDamageEvent event){
if(lastCaster == null) return;
if(event.getEntity().getUniqueId().equals(lastCaster)) event.setCancelled(true);
}
/**
* Gets all Entities in a radius of the location.
*
* Stolen from: 'https://forums.bukkit.org/threads/getnearbyentities-of-a-location.101499/#post-1341141'
*
* @param location
* @param radius
* @return set of all Entities near
*/
public static List<LivingEntity> getNearbyEntities(Location location, int radius){
int chunkRadius = radius < 16 ? 1 : (radius - (radius % 16))/16;
List<LivingEntity> radiusEntities = new LinkedList<LivingEntity>();
for (int chX = 0 -chunkRadius; chX <= chunkRadius; chX ++){
for (int chZ = 0 -chunkRadius; chZ <= chunkRadius; chZ++){
int x=(int) location.getX(),y=(int) location.getY(),z=(int) location.getZ();
for (Entity e : new Location(location.getWorld(),x+(chX*16),y,z+(chZ*16)).getChunk().getEntities()){
if(!(e instanceof LivingEntity)) continue;
if (e.getLocation().distance(location) <= radius && e.getLocation().getBlock() != location.getBlock()){
radiusEntities.add((LivingEntity) e);
}
}
}
}
return radiusEntities;
}
}