Lot of comments

This commit is contained in:
Felix Cravic 2020-05-24 19:59:50 +02:00
parent c2d1a4f70a
commit b193c5f12a
8 changed files with 190 additions and 14 deletions

View File

@ -490,6 +490,7 @@ public abstract class Entity implements Viewable, EventHandler, DataContainer {
} }
public void addPassenger(Entity entity) { public void addPassenger(Entity entity) {
Check.notNull(entity, "Passenger cannot be null");
Check.stateCondition(instance == null, "You need to set an instance using Entity#setInstance"); Check.stateCondition(instance == null, "You need to set an instance using Entity#setInstance");
if (entity.getVehicle() != null) { if (entity.getVehicle() != null) {
@ -503,11 +504,11 @@ public abstract class Entity implements Viewable, EventHandler, DataContainer {
} }
public void removePassenger(Entity entity) { public void removePassenger(Entity entity) {
Check.notNull(entity, "Passenger cannot be null");
Check.stateCondition(instance == null, "You need to set an instance using Entity#setInstance"); Check.stateCondition(instance == null, "You need to set an instance using Entity#setInstance");
if (!passengers.contains(entity)) if (!passengers.remove(entity))
return; return;
this.passengers.remove(entity);
entity.vehicle = null; entity.vehicle = null;
sendPacketToViewersAndSelf(getPassengersPacket()); sendPacketToViewersAndSelf(getPassengersPacket());
} }
@ -516,6 +517,9 @@ public abstract class Entity implements Viewable, EventHandler, DataContainer {
return !passengers.isEmpty(); return !passengers.isEmpty();
} }
/**
* @return an unmodifiable list containing all the entity passengers
*/
public Set<Entity> getPassengers() { public Set<Entity> getPassengers() {
return Collections.unmodifiableSet(passengers); return Collections.unmodifiableSet(passengers);
} }
@ -534,6 +538,11 @@ public abstract class Entity implements Viewable, EventHandler, DataContainer {
return passengersPacket; return passengersPacket;
} }
/**
* Entity statuses can be find <a href="https://wiki.vg/Entity_statuses">here</a>
*
* @param status the status to trigger
*/
public void triggerStatus(byte status) { public void triggerStatus(byte status) {
EntityStatusPacket statusPacket = new EntityStatusPacket(); EntityStatusPacket statusPacket = new EntityStatusPacket();
statusPacket.entityId = getEntityId(); statusPacket.entityId = getEntityId();
@ -541,33 +550,65 @@ public abstract class Entity implements Viewable, EventHandler, DataContainer {
sendPacketToViewersAndSelf(statusPacket); sendPacketToViewersAndSelf(statusPacket);
} }
/**
* @return true if the entity is in fire, false otherwise
*/
public boolean isOnFire() {
return onFire;
}
/**
* Set the entity in fire visually
* <p>
* WARNING: if you want to apply damage or specify a duration,
* see {@link LivingEntity#setFireForDuration(int, TimeUnit)}
*
* @param fire should the entity be set in fire
*/
public void setOnFire(boolean fire) { public void setOnFire(boolean fire) {
this.onFire = fire; this.onFire = fire;
sendMetadataIndex(0); sendMetadataIndex(0);
} }
public boolean isOnFire() {
return onFire;
}
public void setGlowing(boolean glowing) { public void setGlowing(boolean glowing) {
this.glowing = glowing; this.glowing = glowing;
sendMetadataIndex(0); sendMetadataIndex(0);
} }
/**
* @return true if the entity is glowing, false otherwise
*/
public boolean isGlowing() { public boolean isGlowing() {
return glowing; return glowing;
} }
/**
* @param noGravity should the entity ignore gravity
*/
public void setNoGravity(boolean noGravity) { public void setNoGravity(boolean noGravity) {
this.noGravity = noGravity; this.noGravity = noGravity;
sendMetadataIndex(5); sendMetadataIndex(5);
} }
/**
* @return true if the entity ignore gravity, false otherwise
*/
public boolean hasNoGravity() { public boolean hasNoGravity() {
return noGravity; return noGravity;
} }
/**
* Used to refresh the entity and its passengers position
* - put the entity in the right instance chunk
* - update the viewable chunks (load and unload)
* - add/remove players from the viewers list if {@link #isAutoViewable()} is enabled
* <p>
* WARNING: unsafe, should only be used internally in Minestom
*
* @param x new position X
* @param y new position Y
* @param z new position Z
*/
public void refreshPosition(float x, float y, float z) { public void refreshPosition(float x, float y, float z) {
this.lastX = position.getX(); this.lastX = position.getX();
this.lastY = position.getY(); this.lastY = position.getY();

View File

@ -66,18 +66,21 @@ public abstract class EntityCreature extends LivingEntity {
} }
/**
* @param x X movement offset
* @param y Y movement offset
* @param z Z movement offset
* @param updateView should the entity move its head toward the position?
*/
public void move(float x, float y, float z, boolean updateView) { public void move(float x, float y, float z, boolean updateView) {
Position position = getPosition(); Position position = getPosition();
float newX = position.getX() + x;
float newY = position.getY() + y;
float newZ = position.getZ() + z;
Position newPosition = new Position(); Position newPosition = new Position();
// Calculate collisions boxes // Calculate collisions boxes
onGround = CollisionUtils.handlePhysics(this, new Vector(x, y, z), newPosition, new Vector()); onGround = CollisionUtils.handlePhysics(this, new Vector(x, y, z), newPosition, new Vector());
// Refresh target position // Refresh target position
newX = newPosition.getX(); float newX = newPosition.getX();
newY = newPosition.getY(); float newY = newPosition.getY();
newZ = newPosition.getZ(); float newZ = newPosition.getZ();
// Creatures cannot move in unload chunk // Creatures cannot move in unload chunk
if (ChunkUtils.isChunkUnloaded(getInstance(), newX, newZ)) if (ChunkUtils.isChunkUnloaded(getInstance(), newX, newZ))
@ -257,6 +260,13 @@ public abstract class EntityCreature extends LivingEntity {
setPathTo(position, 1000, null); setPathTo(position, 1000, null);
} }
/**
* Used to move the entity toward {@code direction} in the axis X and Z
* Gravity is still applied but the entity will not attempt to jump
*
* @param direction the targeted position
* @param speed define how far the entity will move
*/
public void moveTowards(Position direction, float speed) { public void moveTowards(Position direction, float speed) {
float radians = (float) Math.atan2(direction.getZ() - position.getZ(), direction.getX() - position.getX()); float radians = (float) Math.atan2(direction.getZ() - position.getZ(), direction.getX() - position.getX());
float speedX = (float) (Math.cos(radians) * speed); float speedX = (float) (Math.cos(radians) * speed);

View File

@ -17,6 +17,7 @@ public class ItemEntity extends ObjectEntity {
private boolean pickable = true; private boolean pickable = true;
private boolean mergeable = true; private boolean mergeable = true;
private float mergeRange = 1;
private long spawnTime; private long spawnTime;
private long pickupDelay; private long pickupDelay;
@ -45,7 +46,7 @@ public class ItemEntity extends ObjectEntity {
continue; continue;
// Too far, do not merge // Too far, do not merge
if (getDistance(itemEntity) > 1) if (getDistance(itemEntity) > mergeRange)
continue; continue;
synchronized (this) { synchronized (this) {
@ -125,22 +126,53 @@ public class ItemEntity extends ObjectEntity {
this.pickable = pickable; this.pickable = pickable;
} }
/**
* @return true if the entity is mergeable, false otherwise
*/
public boolean isMergeable() { public boolean isMergeable() {
return mergeable; return mergeable;
} }
/**
* When set to true, close {@link ItemEntity} will try to merge together as a single entity
* when their {@link #getItemStack()} is similar and allowed to stack together
*
* @param mergeable should the entity merge with other {@link ItemEntity}
*/
public void setMergeable(boolean mergeable) { public void setMergeable(boolean mergeable) {
this.mergeable = mergeable; this.mergeable = mergeable;
} }
public float getMergeRange() {
return mergeRange;
}
public void setMergeRange(float mergeRange) {
this.mergeRange = mergeRange;
}
/**
* @return the pickup delay in milliseconds, defined by {@link #setPickupDelay(long, TimeUnit)}
*/
public long getPickupDelay() { public long getPickupDelay() {
return pickupDelay; return pickupDelay;
} }
/**
* Set the pickup delay of the ItemEntity
*
* @param delay
* @param timeUnit
*/
public void setPickupDelay(long delay, TimeUnit timeUnit) { public void setPickupDelay(long delay, TimeUnit timeUnit) {
this.pickupDelay = timeUnit.toMilliseconds(delay); this.pickupDelay = timeUnit.toMilliseconds(delay);
} }
/**
* Used to know if the ItemEntity can be pickup
*
* @return the time in milliseconds since this entity has spawn
*/
public long getSpawnTime() { public long getSpawnTime() {
return spawnTime; return spawnTime;
} }

View File

@ -255,14 +255,30 @@ public abstract class LivingEntity extends Entity implements EquipmentHandler {
return getAttributeValue(Attribute.MAX_HEALTH); return getAttributeValue(Attribute.MAX_HEALTH);
} }
/**
* Set the heal of the entity as its max health
* retrieved from {@link #getAttributeValue(Attribute)} with the attribute {@link Attribute#MAX_HEALTH}
*/
public void heal() { public void heal() {
setHealth(getAttributeValue(Attribute.MAX_HEALTH)); setHealth(getAttributeValue(Attribute.MAX_HEALTH));
} }
/**
* Change the specified attribute value to {@code value}
*
* @param attribute The attribute to change
* @param value the new value of the attribute
*/
public void setAttribute(Attribute attribute, float value) { public void setAttribute(Attribute attribute, float value) {
this.attributeValues[attribute.ordinal()] = value; this.attributeValues[attribute.ordinal()] = value;
} }
/**
* Retrieve the attribute value set by {@link #setAttribute(Attribute, float)}
*
* @param attribute the attribute value to get
* @return the attribute value
*/
public float getAttributeValue(Attribute attribute) { public float getAttributeValue(Attribute attribute) {
return this.attributeValues[attribute.ordinal()]; return this.attributeValues[attribute.ordinal()];
} }
@ -299,10 +315,18 @@ public abstract class LivingEntity extends Entity implements EquipmentHandler {
return isDead; return isDead;
} }
/**
* @return true if the entity is able to pickup items
*/
public boolean canPickupItem() { public boolean canPickupItem() {
return canPickupItem; return canPickupItem;
} }
/**
* When set to false, the entity will not be able to pick {@link ItemEntity} on the ground
*
* @param canPickupItem can the entity pickup item
*/
public void setCanPickupItem(boolean canPickupItem) { public void setCanPickupItem(boolean canPickupItem) {
this.canPickupItem = canPickupItem; this.canPickupItem = canPickupItem;
} }

View File

@ -11,6 +11,11 @@ public abstract class ObjectEntity extends Entity {
setGravity(0.02f); setGravity(0.02f);
} }
/**
* Objects data can be found <a href="https://wiki.vg/Object_Data">here</a>
*
* @return an object data
*/
public abstract int getObjectData(); public abstract int getObjectData();
@Override @Override

View File

@ -797,10 +797,16 @@ public class Player extends LivingEntity {
return playerConnection; return playerConnection;
} }
/**
* @return true if the player is online, false otherwise
*/
public boolean isOnline() { public boolean isOnline() {
return playerConnection.isOnline(); return playerConnection.isOnline();
} }
/**
* @return the player settings
*/
public PlayerSettings getSettings() { public PlayerSettings getSettings() {
return settings; return settings;
} }
@ -809,14 +815,26 @@ public class Player extends LivingEntity {
return inventory; return inventory;
} }
/**
* Used to get the player latency,
* computed by seeing how long it takes the client to answer the {@link KeepAlivePacket} packet
*
* @return the player latency
*/
public int getLatency() { public int getLatency() {
return latency; return latency;
} }
/**
* @return the player current dimension
*/
public Dimension getDimension() { public Dimension getDimension() {
return dimension; return dimension;
} }
/**
* @return the player current gamemode
*/
public GameMode getGameMode() { public GameMode getGameMode() {
return gameMode; return gameMode;
} }
@ -1091,19 +1109,31 @@ public class Player extends LivingEntity {
refreshAbilities(); refreshAbilities();
} }
/**
* @return true if the player if flying, false otherwise
*/
public boolean isFlying() { public boolean isFlying() {
return flying; return flying;
} }
/**
* @param flying should the player fly
*/
public void setFlying(boolean flying) { public void setFlying(boolean flying) {
this.flying = flying; this.flying = flying;
refreshAbilities(); refreshAbilities();
} }
/**
* @return true if the player if allowed to fly, false otherwise
*/
public boolean isAllowFlying() { public boolean isAllowFlying() {
return allowFlying; return allowFlying;
} }
/**
* @param allowFlying should the player be allowed to fly
*/
public void setAllowFlying(boolean allowFlying) { public void setAllowFlying(boolean allowFlying) {
this.allowFlying = allowFlying; this.allowFlying = allowFlying;
refreshAbilities(); refreshAbilities();
@ -1113,6 +1143,14 @@ public class Player extends LivingEntity {
return instantBreak; return instantBreak;
} }
/**
* Change the player ability "Creative Mode"
* <a href="https://wiki.vg/Protocol#Player_Abilities_.28clientbound.29">see</a>
* <p>
* WARNING: this has nothing to do with {@link CustomBlock#getBreakDelay(Player, BlockPosition)}
*
* @param instantBreak
*/
public void setInstantBreak(boolean instantBreak) { public void setInstantBreak(boolean instantBreak) {
this.instantBreak = instantBreak; this.instantBreak = instantBreak;
refreshAbilities(); refreshAbilities();

View File

@ -8,8 +8,10 @@ public class PlayerHeldListener {
public static void heldListener(ClientHeldItemChangePacket packet, Player player) { public static void heldListener(ClientHeldItemChangePacket packet, Player player) {
short slot = packet.slot; short slot = packet.slot;
if (!MathUtils.isBetween(slot, 0, 8)) if (!MathUtils.isBetween(slot, 0, 8)) {
// Incorrect packet, ignore
return; return;
}
player.refreshHeldSlot(slot); player.refreshHeldSlot(slot);
} }

View File

@ -5,6 +5,12 @@ import net.minestom.server.instance.Instance;
public class ChunkUtils { public class ChunkUtils {
/**
* @param instance the instance to check
* @param x instance X coordinate
* @param z instance Z coordinate
* @return true if the chunk is unloaded, false otherwise
*/
public static boolean isChunkUnloaded(Instance instance, float x, float z) { public static boolean isChunkUnloaded(Instance instance, float x, float z) {
int chunkX = getChunkCoordinate((int) x); int chunkX = getChunkCoordinate((int) x);
int chunkZ = getChunkCoordinate((int) z); int chunkZ = getChunkCoordinate((int) z);
@ -13,15 +19,28 @@ public class ChunkUtils {
return chunk == null || !chunk.isLoaded(); return chunk == null || !chunk.isLoaded();
} }
/**
* @param xz the instance coordinate to convert
* @return the chunk X or Z based on the argument
*/
public static int getChunkCoordinate(int xz) { public static int getChunkCoordinate(int xz) {
// Assume Chunk.CHUNK_SIZE_X == Chunk.CHUNK_SIZE_Z // Assume Chunk.CHUNK_SIZE_X == Chunk.CHUNK_SIZE_Z
return Math.floorDiv(xz, Chunk.CHUNK_SIZE_X); return Math.floorDiv(xz, Chunk.CHUNK_SIZE_X);
} }
/**
* @param chunkX the chunk X
* @param chunkZ the chunk Z
* @return a number storing the chunk X and Z
*/
public static long getChunkIndex(int chunkX, int chunkZ) { public static long getChunkIndex(int chunkX, int chunkZ) {
return (((long) chunkX) << 32) | (chunkZ & 0xffffffffL); return (((long) chunkX) << 32) | (chunkZ & 0xffffffffL);
} }
/**
* @param index the chunk index computed by {@link #getChunkIndex(int, int)}
* @return an array containing both the chunk X and Z (index 0 = X; index 1 = Z)
*/
public static int[] getChunkCoord(long index) { public static int[] getChunkCoord(long index) {
int chunkX = (int) (index >> 32); int chunkX = (int) (index >> 32);
int chunkZ = (int) index; int chunkZ = (int) index;
@ -32,6 +51,11 @@ public class ChunkUtils {
return y / Chunk.CHUNK_SECTION_SIZE; return y / Chunk.CHUNK_SECTION_SIZE;
} }
/**
* @param position the initial position
* @param range how far should it retrieves chunk
* @return an array containing chunks index which can be converted using {@link #getChunkCoord(long)}
*/
public static long[] getChunksInRange(final Position position, int range) { public static long[] getChunksInRange(final Position position, int range) {
long[] visibleChunks = new long[MathUtils.square(range + 1)]; long[] visibleChunks = new long[MathUtils.square(range + 1)];
final int startLoop = -(range / 2); final int startLoop = -(range / 2);