| package net.glowstone.entity; |
| |
| import gnu.trove.set.hash.TIntHashSet; |
| import net.glowstone.GlowServer; |
| import net.glowstone.GlowWorld; |
| import net.glowstone.msg.*; |
| import net.glowstone.util.Parameter; |
| import net.glowstone.util.Position; |
| import net.glowstone.util.TargetBlock; |
| import org.bukkit.Location; |
| import org.bukkit.block.Block; |
| import org.bukkit.entity.*; |
| import org.bukkit.inventory.EntityEquipment; |
| import org.bukkit.potion.PotionEffect; |
| import org.bukkit.potion.PotionEffectType; |
| |
| import java.util.*; |
| |
| /** |
| * A GlowLivingEntity is a {@link org.bukkit.entity.Player} or {@link org.bukkit.entity.Monster}. |
| * @author Graham Edgecombe. |
| */ |
| public abstract class GlowLivingEntity extends GlowEntity implements LivingEntity { |
| |
| /** |
| * The entity's health. |
| */ |
| protected int health = 0; |
| |
| /** |
| * The monster's metadata. |
| */ |
| protected final List<Parameter<?>> metadata = new ArrayList<Parameter<?>>(); |
| |
| /** |
| * Potion effects on the entity. |
| */ |
| private final Map<PotionEffectType, PotionEffect> potionEffects = new HashMap<PotionEffectType, PotionEffect>(); |
| |
| /** |
| * Creates a mob within the specified world. |
| * @param world The world. |
| */ |
| public GlowLivingEntity(GlowServer server, GlowWorld world) { |
| super(server, world); |
| } |
| |
| @Override |
| public Message createUpdateMessage() { |
| boolean moved = hasMoved(); |
| boolean rotated = hasRotated(); |
| |
| int x = Position.getIntX(location); |
| int y = Position.getIntY(location); |
| int z = Position.getIntZ(location); |
| |
| int dx = x - Position.getIntX(previousLocation); |
| int dy = y - Position.getIntY(previousLocation); |
| int dz = z - Position.getIntZ(previousLocation); |
| |
| boolean teleport = dx > Byte.MAX_VALUE || dy > Byte.MAX_VALUE || dz > Byte.MAX_VALUE || dx < Byte.MIN_VALUE || dy < Byte.MIN_VALUE || dz < Byte.MIN_VALUE; |
| |
| int yaw = Position.getIntYaw(previousLocation); |
| int pitch = Position.getIntPitch(previousLocation); |
| |
| if (moved && teleport) { |
| return new EntityTeleportMessage(id, x, y, z, yaw, pitch); |
| } else if (moved && rotated) { |
| return new RelativeEntityPositionRotationMessage(id, dx, dy, dz, yaw, pitch); |
| } else if (moved) { |
| return new RelativeEntityPositionMessage(id, dx, dy, dz); |
| } else if (rotated) { |
| return new EntityRotationMessage(id, yaw, pitch); |
| } |
| |
| return null; |
| } |
| |
| public double getHealth() { |
| return health; |
| } |
| |
| public void setHealth(int health) { |
| if (health < 0) health = 0; |
| if (health > 200) health = 200; |
| this.health = health; |
| } |
| |
| public double getEyeHeight() { |
| return getEyeHeight(false); |
| } |
| |
| public double getEyeHeight(boolean ignoreSneaking) { |
| if (false /* TODO: sneaking */ || !ignoreSneaking) { |
| return 1.6; |
| } else { |
| return 1.4; |
| } |
| } |
| |
| public Location getEyeLocation() { |
| Location loc = getLocation(); |
| loc.setY(loc.getY() + getEyeHeight()); |
| return loc; |
| } |
| |
| public List<Block> getLineOfSight(HashSet<Byte> transparent, int maxDistance) { |
| TIntHashSet transparentBlocks = new TIntHashSet(); |
| if (transparent != null) { |
| for (byte byt : transparent) { |
| transparentBlocks.add(byt); |
| } |
| } else { |
| transparentBlocks.add(0); |
| } |
| List<Block> ret = new ArrayList<Block>(); |
| TargetBlock target = new TargetBlock(this, maxDistance, 0.2, transparentBlocks); |
| while (target.getNextBlock()) { |
| Block block = target.getCurrentBlock().getBlock(); |
| if (!transparentBlocks.contains(block.getTypeId())) { |
| ret.add(block); |
| } |
| } |
| return ret; |
| } |
| |
| public Block getTargetBlock(HashSet<Byte> transparent, int maxDistance) { |
| TIntHashSet transparentBlocks = new TIntHashSet(); |
| if (transparent != null) { |
| for (byte byt : transparent) { |
| transparentBlocks.add(byt); |
| } |
| } else { |
| transparentBlocks = null; |
| } |
| Location loc = new TargetBlock(this, maxDistance, 0.2, transparentBlocks).getSolidTargetBlock(); |
| return loc == null ? null : loc.getBlock(); |
| } |
| |
| public List<Block> getLastTwoTargetBlocks(HashSet<Byte> transparent, int maxDistance) { |
| TIntHashSet transparentBlocks = new TIntHashSet(); |
| if (transparent != null) { |
| for (byte byt : transparent) { |
| transparentBlocks.add(byt); |
| } |
| } else { |
| transparentBlocks = null; |
| } |
| TargetBlock target = new TargetBlock(this, maxDistance, 0.2, transparentBlocks); |
| Location last = target.getSolidTargetBlock(); |
| if (last == null) { |
| return new ArrayList<Block>(Arrays.asList(target.getPreviousBlock().getBlock())); |
| } |
| return new ArrayList<Block>(Arrays.asList(target.getPreviousBlock().getBlock(), last.getBlock())); |
| } |
| |
| public Egg throwEgg() { |
| throw new UnsupportedOperationException("Not supported yet."); |
| } |
| |
| public Snowball throwSnowball() { |
| throw new UnsupportedOperationException("Not supported yet."); |
| } |
| |
| public Arrow shootArrow() { |
| throw new UnsupportedOperationException("Not supported yet."); |
| } |
| |
| public int getMaximumNoDamageTicks() { |
| throw new UnsupportedOperationException("Not supported yet."); |
| } |
| |
| public void setMaximumNoDamageTicks(int ticks) { |
| throw new UnsupportedOperationException("Not supported yet."); |
| } |
| |
| public double getLastDamage() { |
| throw new UnsupportedOperationException("Not supported yet."); |
| } |
| |
| public void setLastDamage(int damage) { |
| throw new UnsupportedOperationException("Not supported yet."); |
| } |
| |
| public int getNoDamageTicks() { |
| throw new UnsupportedOperationException("Not supported yet."); |
| } |
| |
| public void setNoDamageTicks(int ticks) { |
| throw new UnsupportedOperationException("Not supported yet."); |
| } |
| |
| protected Parameter<?> getMetadata(int index) { |
| return metadata.get(index); |
| } |
| |
| protected void setMetadata(Parameter<?> data) { |
| if(data.getIndex() < metadata.size()) { |
| metadata.set(data.getIndex(), data); |
| } else { |
| metadata.add(data); |
| } |
| } |
| |
| @Override |
| public void pulse() { |
| super.pulse(); |
| // todo: tick down potion effects |
| } |
| |
| // NEW STUFF |
| |
| @Override |
| public <T extends Projectile> T launchProjectile(Class<? extends T> projectile) { |
| return null; |
| } |
| |
| @Override |
| public int _INVALID_getLastDamage() { |
| return 0; |
| } |
| |
| @Override |
| public void setLastDamage(double damage) { |
| |
| } |
| |
| @Override |
| public void _INVALID_setLastDamage(int damage) { |
| |
| } |
| |
| @Override |
| public Player getKiller() { |
| return null; |
| } |
| |
| @Override |
| public boolean addPotionEffect(PotionEffect effect) { |
| return addPotionEffect(effect, false); |
| } |
| |
| @Override |
| public boolean addPotionEffect(PotionEffect effect, boolean force) { |
| if (potionEffects.containsKey(effect.getType())) { |
| if (force) { |
| removePotionEffect(effect.getType()); |
| } else { |
| return false; |
| } |
| } |
| |
| potionEffects.put(effect.getType(), effect); |
| |
| // todo: this, updated, only players in range |
| /*EntityEffectMessage msg = new EntityEffectMessage(getEntityId(), effect.getType().getId(), effect.getAmplifier(), effect.getDuration()); |
| for (Player player : server.getOnlinePlayers()) { |
| ((GlowPlayer) player).getSession().send(msg); |
| }*/ |
| return true; |
| } |
| |
| @Override |
| public boolean addPotionEffects(Collection<PotionEffect> effects) { |
| boolean result = true; |
| for (PotionEffect effect : effects) { |
| if (!addPotionEffect(effect)) { |
| result = false; |
| } |
| } |
| return result; |
| } |
| |
| @Override |
| public boolean hasPotionEffect(PotionEffectType type) { |
| return potionEffects.containsKey(type); |
| } |
| |
| @Override |
| public void removePotionEffect(PotionEffectType type) { |
| if (!hasPotionEffect(type)) return; |
| potionEffects.remove(type); |
| |
| // todo: this, improved, for players in range |
| /*EntityRemoveEffectMessage msg = new EntityRemoveEffectMessage(getEntityId(), type.getId()); |
| for (Player player : server.getOnlinePlayers()) { |
| ((GlowPlayer) player).getSession().send(msg); |
| }*/ |
| } |
| |
| @Override |
| public Collection<PotionEffect> getActivePotionEffects() { |
| return Collections.unmodifiableCollection(potionEffects.values()); |
| } |
| |
| @Override |
| public boolean hasLineOfSight(Entity other) { |
| return false; |
| } |
| |
| @Override |
| public boolean getRemoveWhenFarAway() { |
| return false; |
| } |
| |
| @Override |
| public void setRemoveWhenFarAway(boolean remove) { |
| |
| } |
| |
| @Override |
| public EntityEquipment getEquipment() { |
| return null; |
| } |
| |
| @Override |
| public void setCanPickupItems(boolean pickup) { |
| |
| } |
| |
| @Override |
| public boolean getCanPickupItems() { |
| return false; |
| } |
| |
| @Override |
| public void setCustomName(String name) { |
| |
| } |
| |
| @Override |
| public String getCustomName() { |
| return null; |
| } |
| |
| @Override |
| public void setCustomNameVisible(boolean flag) { |
| |
| } |
| |
| @Override |
| public boolean isCustomNameVisible() { |
| return false; |
| } |
| |
| @Override |
| public boolean isLeashed() { |
| return false; |
| } |
| |
| @Override |
| public Entity getLeashHolder() throws IllegalStateException { |
| return null; |
| } |
| |
| @Override |
| public boolean setLeashHolder(Entity holder) { |
| return false; |
| } |
| |
| @Override |
| public void damage(double amount) { |
| |
| } |
| |
| @Override |
| public void _INVALID_damage(int amount) { |
| |
| } |
| |
| @Override |
| public void damage(double amount, Entity source) { |
| |
| } |
| |
| @Override |
| public void _INVALID_damage(int amount, Entity source) { |
| |
| } |
| |
| @Override |
| public int _INVALID_getHealth() { |
| return 0; |
| } |
| |
| @Override |
| public void setHealth(double health) { |
| |
| } |
| |
| @Override |
| public void _INVALID_setHealth(int health) { |
| |
| } |
| |
| @Override |
| public double getMaxHealth() { |
| return 0; |
| } |
| |
| @Override |
| public int _INVALID_getMaxHealth() { |
| return 0; |
| } |
| |
| @Override |
| public void setMaxHealth(double health) { |
| |
| } |
| |
| @Override |
| public void _INVALID_setMaxHealth(int health) { |
| |
| } |
| |
| @Override |
| public void resetMaxHealth() { |
| |
| } |
| } |