2012-04-27 11:47:11 +02:00
|
|
|
package com.gmail.nossr50.util;
|
|
|
|
|
|
|
|
import java.util.Random;
|
|
|
|
|
|
|
|
import org.bukkit.Location;
|
|
|
|
import org.bukkit.Material;
|
|
|
|
import org.bukkit.block.Block;
|
2013-01-14 17:55:43 +01:00
|
|
|
import org.bukkit.entity.AnimalTamer;
|
2013-01-24 21:46:29 +01:00
|
|
|
import org.bukkit.entity.Entity;
|
2013-01-03 03:03:52 +01:00
|
|
|
import org.bukkit.entity.Item;
|
2012-04-27 11:47:11 +02:00
|
|
|
import org.bukkit.entity.LivingEntity;
|
|
|
|
import org.bukkit.entity.Player;
|
2013-01-14 17:55:43 +01:00
|
|
|
import org.bukkit.entity.Tameable;
|
2012-04-27 11:47:11 +02:00
|
|
|
import org.bukkit.event.entity.EntityDamageEvent;
|
|
|
|
import org.bukkit.inventory.ItemStack;
|
2012-06-04 14:51:10 +02:00
|
|
|
import org.bukkit.plugin.PluginManager;
|
2012-04-27 11:47:11 +02:00
|
|
|
|
2012-06-07 00:02:22 +02:00
|
|
|
import com.gmail.nossr50.mcMMO;
|
2013-01-15 22:16:46 +01:00
|
|
|
import com.gmail.nossr50.config.AdvancedConfig;
|
2012-04-27 11:47:11 +02:00
|
|
|
import com.gmail.nossr50.config.Config;
|
|
|
|
import com.gmail.nossr50.events.fake.FakeBlockBreakEvent;
|
2012-06-03 23:38:33 +02:00
|
|
|
import com.gmail.nossr50.events.fake.FakeBlockDamageEvent;
|
2012-04-27 11:47:11 +02:00
|
|
|
import com.gmail.nossr50.events.fake.FakePlayerAnimationEvent;
|
|
|
|
import com.gmail.nossr50.events.items.McMMOItemSpawnEvent;
|
2013-01-30 17:35:33 +01:00
|
|
|
import com.gmail.nossr50.mods.ModChecks;
|
2013-01-14 17:55:43 +01:00
|
|
|
import com.gmail.nossr50.party.PartyManager;
|
2012-04-27 11:47:11 +02:00
|
|
|
|
2013-01-26 23:01:55 +01:00
|
|
|
public final class Misc {
|
2012-04-27 11:47:11 +02:00
|
|
|
private static Random random = new Random();
|
2013-01-15 22:16:46 +01:00
|
|
|
public static int toolDurabilityLoss = Config.getInstance().getAbilityToolDamage();
|
|
|
|
public static int abilityLengthIncreaseLevel = AdvancedConfig.getInstance().getAbilityLength();
|
2013-01-24 21:46:29 +01:00
|
|
|
public static boolean isSpawnerXPEnabled = Config.getInstance().getExperienceGainsMobspawnersEnabled();
|
2013-01-08 22:07:29 +01:00
|
|
|
public static final int PLAYER_RESPAWN_COOLDOWN_SECONDS = 5;
|
2013-01-10 03:44:53 +01:00
|
|
|
public static final int TIME_CONVERSION_FACTOR = 1000;
|
|
|
|
public static final double SKILL_MESSAGE_MAX_SENDING_DISTANCE = 10.0;
|
2013-01-30 00:36:16 +01:00
|
|
|
//Sound Pitches & Volumes from CB
|
|
|
|
public static final float ANVIL_USE_PITCH = 0.3F; // Not in CB directly, I went off the place sound values
|
|
|
|
public static final float ANVIL_USE_VOLUME = 1.0F; // Not in CB directly, I went off the place sound values
|
|
|
|
public static final float FIZZ_PITCH = 2.6F + (Misc.getRandom().nextFloat() - Misc.getRandom().nextFloat()) * 0.8F;
|
|
|
|
public static final float FIZZ_VOLUME = 0.5F;
|
|
|
|
public static final float POP_PITCH = ((getRandom().nextFloat() - getRandom().nextFloat()) * 0.7F + 1.0F) * 2.0F;
|
|
|
|
public static final float POP_VOLUME = 0.2F;
|
|
|
|
|
2013-01-26 23:01:55 +01:00
|
|
|
private Misc() {};
|
|
|
|
|
2013-01-14 17:55:43 +01:00
|
|
|
public static boolean isFriendlyPet(Player attacker, Tameable pet) {
|
|
|
|
if (pet.isTamed()) {
|
|
|
|
AnimalTamer tamer = pet.getOwner();
|
|
|
|
|
|
|
|
if (tamer instanceof Player) {
|
|
|
|
Player owner = (Player) tamer;
|
|
|
|
|
2013-01-26 23:01:55 +01:00
|
|
|
if (owner == attacker || PartyManager.inSameParty(attacker, owner)) {
|
2013-01-14 17:55:43 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-01-24 21:46:29 +01:00
|
|
|
public static boolean isNPCEntity(Entity entity) {
|
|
|
|
if (entity == null || entity.hasMetadata("NPC")) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static boolean isNPCPlayer(Player player) {
|
2013-02-03 13:27:47 +01:00
|
|
|
if (player == null || player.hasMetadata("NPC")) {
|
2013-01-24 17:45:40 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-04-27 11:47:11 +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;
|
|
|
|
}
|
2013-01-10 04:43:21 +01:00
|
|
|
|
|
|
|
return false;
|
2012-04-27 11:47:11 +02:00
|
|
|
}
|
2012-06-05 15:57:10 +02:00
|
|
|
|
2012-04-27 11:47:11 +02:00
|
|
|
/**
|
|
|
|
* Simulate a block break event.
|
|
|
|
*
|
|
|
|
* @param block The block to break
|
|
|
|
* @param player The player breaking the block
|
|
|
|
* @param shouldArmSwing true if an armswing event should be fired, false otherwise
|
|
|
|
* @return true if the event wasn't cancelled, false otherwise
|
|
|
|
*/
|
|
|
|
public static boolean blockBreakSimulate(Block block, Player player, Boolean shouldArmSwing) {
|
|
|
|
|
|
|
|
//Support for NoCheat
|
|
|
|
if (shouldArmSwing) {
|
|
|
|
FakePlayerAnimationEvent armswing = new FakePlayerAnimationEvent(player);
|
2012-06-07 00:02:22 +02:00
|
|
|
mcMMO.p.getServer().getPluginManager().callEvent(armswing);
|
2012-04-27 11:47:11 +02:00
|
|
|
}
|
2012-06-04 14:51:10 +02:00
|
|
|
|
2012-06-07 00:02:22 +02:00
|
|
|
PluginManager pluginManger = mcMMO.p.getServer().getPluginManager();
|
2012-06-04 14:51:10 +02:00
|
|
|
|
2012-06-03 23:38:33 +02:00
|
|
|
FakeBlockDamageEvent damageEvent = new FakeBlockDamageEvent(player, block, player.getItemInHand(), true);
|
2012-06-04 14:51:10 +02:00
|
|
|
pluginManger.callEvent(damageEvent);
|
|
|
|
|
2012-06-03 23:38:33 +02:00
|
|
|
FakeBlockBreakEvent breakEvent = new FakeBlockBreakEvent(block, player);
|
2012-06-04 14:51:10 +02:00
|
|
|
pluginManger.callEvent(breakEvent);
|
2012-04-27 11:47:11 +02:00
|
|
|
|
2012-11-19 05:25:36 +01:00
|
|
|
if (!damageEvent.isCancelled() && !breakEvent.isCancelled()) {
|
2012-04-27 11:47:11 +02:00
|
|
|
return true;
|
2012-06-04 14:51:10 +02:00
|
|
|
}
|
2013-01-10 04:43:21 +01:00
|
|
|
|
|
|
|
return false;
|
2012-04-27 11:47:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the upgrade tier of the item in hand.
|
|
|
|
*
|
|
|
|
* @param inHand The item to check the tier of
|
|
|
|
* @return the tier of the item
|
|
|
|
*/
|
|
|
|
public static int getTier(ItemStack inHand) {
|
|
|
|
int tier = 0;
|
|
|
|
|
|
|
|
if (ItemChecks.isWoodTool(inHand)) {
|
|
|
|
tier = 1;
|
|
|
|
}
|
|
|
|
else if (ItemChecks.isStoneTool(inHand)) {
|
|
|
|
tier = 2;
|
|
|
|
}
|
|
|
|
else if (ItemChecks.isIronTool(inHand)) {
|
|
|
|
tier = 3;
|
|
|
|
}
|
2012-05-18 03:19:17 +02:00
|
|
|
else if (ItemChecks.isGoldTool(inHand)) {
|
2012-04-27 11:47:11 +02:00
|
|
|
tier = 1;
|
|
|
|
}
|
2012-05-18 03:19:17 +02:00
|
|
|
else if (ItemChecks.isDiamondTool(inHand)) {
|
2012-04-27 11:47:11 +02:00
|
|
|
tier = 4;
|
2012-05-18 03:19:17 +02:00
|
|
|
}
|
2012-05-18 20:29:53 +02:00
|
|
|
else if (ModChecks.isCustomTool(inHand)) {
|
2012-05-18 03:19:17 +02:00
|
|
|
tier = ModChecks.getToolFromItemStack(inHand).getTier();
|
|
|
|
}
|
2012-04-27 11:47:11 +02:00
|
|
|
|
|
|
|
return tier;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine if two locations are near each other.
|
|
|
|
*
|
|
|
|
* @param first The first location
|
|
|
|
* @param second The second location
|
|
|
|
* @param maxDistance The max distance apart
|
|
|
|
* @return true if the distance between <code>first</code> and <code>second</code> is less than <code>maxDistance</code>, false otherwise
|
|
|
|
*/
|
|
|
|
public static boolean isNear(Location first, Location second, double maxDistance) {
|
|
|
|
if (!first.getWorld().equals(second.getWorld())) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (first.distanceSquared(second) < (maxDistance * maxDistance)) {
|
|
|
|
return true;
|
|
|
|
}
|
2013-01-10 04:43:21 +01:00
|
|
|
|
|
|
|
return false;
|
2012-04-27 11:47:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Drop items at a given location.
|
|
|
|
*
|
|
|
|
* @param location The location to drop the items at
|
|
|
|
* @param is The items to drop
|
|
|
|
* @param quantity The amount of items to drop
|
|
|
|
*/
|
2012-05-18 19:40:21 +02:00
|
|
|
public static void dropItems(Location location, ItemStack is, int quantity) {
|
2012-04-27 11:47:11 +02:00
|
|
|
for (int i = 0; i < quantity; i++) {
|
2012-05-18 19:40:21 +02:00
|
|
|
dropItem(location, is);
|
2012-04-27 11:47:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Randomly drop an item at a given location.
|
|
|
|
*
|
|
|
|
* @param location The location to drop the items at
|
|
|
|
* @param is The item to drop
|
|
|
|
* @param chance The percentage chance for the item to drop
|
|
|
|
*/
|
2012-11-13 04:10:14 +01:00
|
|
|
public static void randomDropItem(Location location, ItemStack is, int chance) {
|
2012-04-27 11:47:11 +02:00
|
|
|
if (random.nextInt(100) < chance) {
|
2012-05-18 19:40:21 +02:00
|
|
|
dropItem(location, is);
|
2012-04-27 11:47:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Randomly drop items at a given location.
|
|
|
|
*
|
|
|
|
* @param location The location to drop the items at
|
|
|
|
* @param is The item to drop
|
|
|
|
* @param chance The percentage chance for the item to drop
|
|
|
|
* @param quantity The amount of items to drop
|
|
|
|
*/
|
2012-05-18 19:40:21 +02:00
|
|
|
public static void randomDropItems(Location location, ItemStack is, int chance, int quantity) {
|
2013-02-16 23:29:44 +01:00
|
|
|
int dropCount = random.nextInt(quantity);
|
|
|
|
|
|
|
|
//I could just have the itemstacks quantity value changed but I think this will make it look more natural
|
|
|
|
for (int i = 0; i < dropCount; i++) {
|
|
|
|
dropItem(location, is);
|
|
|
|
}
|
2012-04-27 11:47:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Drop an item at a given location.
|
|
|
|
*
|
|
|
|
* @param location The location to drop the item at
|
|
|
|
* @param itemStack The item to drop
|
|
|
|
*/
|
2012-05-18 19:40:21 +02:00
|
|
|
public static void dropItem(Location location, ItemStack itemStack) {
|
2012-04-27 11:47:11 +02:00
|
|
|
|
2013-02-03 09:12:34 +01:00
|
|
|
if (itemStack.getType() == Material.AIR) {
|
2013-01-14 20:31:11 +01:00
|
|
|
return;
|
2013-02-01 20:58:03 +01:00
|
|
|
}
|
2013-01-14 20:31:11 +01:00
|
|
|
|
2012-04-27 11:47:11 +02:00
|
|
|
// We can't get the item until we spawn it and we want to make it cancellable, so we have a custom event.
|
|
|
|
McMMOItemSpawnEvent event = new McMMOItemSpawnEvent(location, itemStack);
|
2012-06-07 00:02:22 +02:00
|
|
|
mcMMO.p.getServer().getPluginManager().callEvent(event);
|
2012-04-27 11:47:11 +02:00
|
|
|
|
2013-01-14 20:31:11 +01:00
|
|
|
if (event.isCancelled())
|
2012-04-27 11:47:11 +02:00
|
|
|
return;
|
|
|
|
|
2013-01-03 03:03:52 +01:00
|
|
|
Item newItem = location.getWorld().dropItemNaturally(location, itemStack);
|
|
|
|
|
|
|
|
ItemStack cloned = itemStack.clone();
|
|
|
|
cloned.setAmount(newItem.getItemStack().getAmount());
|
|
|
|
|
|
|
|
newItem.setItemStack(cloned);
|
2012-04-27 11:47:11 +02:00
|
|
|
}
|
|
|
|
|
2012-06-04 15:03:25 +02:00
|
|
|
/**
|
|
|
|
* Get the max power level for a player.
|
|
|
|
*
|
|
|
|
* @return the maximum power level for a player
|
|
|
|
*/
|
2012-04-27 11:47:11 +02:00
|
|
|
public static int getPowerLevelCap() {
|
2012-05-21 13:53:52 +02:00
|
|
|
int levelCap = Config.getInstance().getPowerLevelCap();
|
|
|
|
|
|
|
|
if (levelCap > 0) {
|
|
|
|
return levelCap;
|
2012-04-27 11:47:11 +02:00
|
|
|
}
|
2013-01-10 04:43:21 +01:00
|
|
|
|
|
|
|
return Integer.MAX_VALUE;
|
2012-04-27 11:47:11 +02:00
|
|
|
}
|
2013-01-10 05:46:35 +01:00
|
|
|
|
|
|
|
public static Random getRandom() {
|
|
|
|
return random;
|
|
|
|
}
|
2012-04-27 11:47:11 +02:00
|
|
|
}
|