mirror of
https://github.com/Minestom/Minestom.git
synced 2025-01-08 01:17:47 +01:00
chore: port DamageType cleanup change from MelonHell 1.19.4 PR
This commit is contained in:
parent
d502a4fa62
commit
9eab3d4f8b
@ -33,6 +33,7 @@ public class Generators {
|
|||||||
generator.generate(resource("particles.json"), "net.minestom.server.particle", "Particle", "ParticleImpl", "Particles");
|
generator.generate(resource("particles.json"), "net.minestom.server.particle", "Particle", "ParticleImpl", "Particles");
|
||||||
generator.generate(resource("sounds.json"), "net.minestom.server.sound", "SoundEvent", "SoundEventImpl", "SoundEvents");
|
generator.generate(resource("sounds.json"), "net.minestom.server.sound", "SoundEvent", "SoundEventImpl", "SoundEvents");
|
||||||
generator.generate(resource("custom_statistics.json"), "net.minestom.server.statistic", "StatisticType", "StatisticTypeImpl", "StatisticTypes");
|
generator.generate(resource("custom_statistics.json"), "net.minestom.server.statistic", "StatisticType", "StatisticTypeImpl", "StatisticTypes");
|
||||||
|
generator.generate(resource("damage_types.json"), "net.minestom.server.entity.damage", "DamageType", "DamageTypeImpl", "DamageTypes");
|
||||||
|
|
||||||
// Generate fluids
|
// Generate fluids
|
||||||
new FluidGenerator(resource("fluids.json"), outputFolder).generate();
|
new FluidGenerator(resource("fluids.json"), outputFolder).generate();
|
||||||
|
@ -10,7 +10,7 @@ import net.minestom.server.entity.Entity;
|
|||||||
import net.minestom.server.entity.GameMode;
|
import net.minestom.server.entity.GameMode;
|
||||||
import net.minestom.server.entity.ItemEntity;
|
import net.minestom.server.entity.ItemEntity;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.entity.damage.DamageType;
|
import net.minestom.server.entity.damage.Damage;
|
||||||
import net.minestom.server.event.Event;
|
import net.minestom.server.event.Event;
|
||||||
import net.minestom.server.event.EventNode;
|
import net.minestom.server.event.EventNode;
|
||||||
import net.minestom.server.event.entity.EntityAttackEvent;
|
import net.minestom.server.event.entity.EntityAttackEvent;
|
||||||
@ -28,13 +28,11 @@ import net.minestom.server.item.metadata.BundleMeta;
|
|||||||
import net.minestom.server.monitoring.BenchmarkManager;
|
import net.minestom.server.monitoring.BenchmarkManager;
|
||||||
import net.minestom.server.monitoring.TickMonitor;
|
import net.minestom.server.monitoring.TickMonitor;
|
||||||
import net.minestom.server.utils.MathUtils;
|
import net.minestom.server.utils.MathUtils;
|
||||||
import net.minestom.server.utils.chunk.ChunkUtils;
|
|
||||||
import net.minestom.server.utils.time.TimeUnit;
|
import net.minestom.server.utils.time.TimeUnit;
|
||||||
import net.minestom.server.world.DimensionType;
|
import net.minestom.server.world.DimensionType;
|
||||||
|
|
||||||
import java.time.Duration;
|
import java.time.Duration;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
import java.util.concurrent.CompletableFuture;
|
|
||||||
import java.util.concurrent.ThreadLocalRandom;
|
import java.util.concurrent.ThreadLocalRandom;
|
||||||
import java.util.concurrent.atomic.AtomicReference;
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
|
||||||
@ -51,7 +49,7 @@ public class PlayerInit {
|
|||||||
|
|
||||||
if (entity instanceof Player) {
|
if (entity instanceof Player) {
|
||||||
Player target = (Player) entity;
|
Player target = (Player) entity;
|
||||||
target.damage(DamageType.fromEntity(source), 5);
|
target.damage(Damage.fromEntity(source, 5));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (source instanceof Player) {
|
if (source instanceof Player) {
|
||||||
|
@ -0,0 +1,95 @@
|
|||||||
|
package net.minestom.server.entity.damage;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Code autogenerated, do not edit!
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
interface DamageTypes {
|
||||||
|
DamageType WITHER = DamageTypeImpl.get("minecraft:wither");
|
||||||
|
|
||||||
|
DamageType SONIC_BOOM = DamageTypeImpl.get("minecraft:sonic_boom");
|
||||||
|
|
||||||
|
DamageType WITHER_SKULL = DamageTypeImpl.get("minecraft:wither_skull");
|
||||||
|
|
||||||
|
DamageType DRY_OUT = DamageTypeImpl.get("minecraft:dry_out");
|
||||||
|
|
||||||
|
DamageType TRIDENT = DamageTypeImpl.get("minecraft:trident");
|
||||||
|
|
||||||
|
DamageType ON_FIRE = DamageTypeImpl.get("minecraft:on_fire");
|
||||||
|
|
||||||
|
DamageType FALL = DamageTypeImpl.get("minecraft:fall");
|
||||||
|
|
||||||
|
DamageType MOB_ATTACK = DamageTypeImpl.get("minecraft:mob_attack");
|
||||||
|
|
||||||
|
DamageType MOB_PROJECTILE = DamageTypeImpl.get("minecraft:mob_projectile");
|
||||||
|
|
||||||
|
DamageType THROWN = DamageTypeImpl.get("minecraft:thrown");
|
||||||
|
|
||||||
|
DamageType FALLING_STALACTITE = DamageTypeImpl.get("minecraft:falling_stalactite");
|
||||||
|
|
||||||
|
DamageType FIREBALL = DamageTypeImpl.get("minecraft:fireball");
|
||||||
|
|
||||||
|
DamageType FALLING_BLOCK = DamageTypeImpl.get("minecraft:falling_block");
|
||||||
|
|
||||||
|
DamageType PLAYER_EXPLOSION = DamageTypeImpl.get("minecraft:player_explosion");
|
||||||
|
|
||||||
|
DamageType STING = DamageTypeImpl.get("minecraft:sting");
|
||||||
|
|
||||||
|
DamageType UNATTRIBUTED_FIREBALL = DamageTypeImpl.get("minecraft:unattributed_fireball");
|
||||||
|
|
||||||
|
DamageType IN_WALL = DamageTypeImpl.get("minecraft:in_wall");
|
||||||
|
|
||||||
|
DamageType IN_FIRE = DamageTypeImpl.get("minecraft:in_fire");
|
||||||
|
|
||||||
|
DamageType ARROW = DamageTypeImpl.get("minecraft:arrow");
|
||||||
|
|
||||||
|
DamageType HOT_FLOOR = DamageTypeImpl.get("minecraft:hot_floor");
|
||||||
|
|
||||||
|
DamageType DROWN = DamageTypeImpl.get("minecraft:drown");
|
||||||
|
|
||||||
|
DamageType STARVE = DamageTypeImpl.get("minecraft:starve");
|
||||||
|
|
||||||
|
DamageType GENERIC_KILL = DamageTypeImpl.get("minecraft:generic_kill");
|
||||||
|
|
||||||
|
DamageType DRAGON_BREATH = DamageTypeImpl.get("minecraft:dragon_breath");
|
||||||
|
|
||||||
|
DamageType MOB_ATTACK_NO_AGGRO = DamageTypeImpl.get("minecraft:mob_attack_no_aggro");
|
||||||
|
|
||||||
|
DamageType LAVA = DamageTypeImpl.get("minecraft:lava");
|
||||||
|
|
||||||
|
DamageType OUTSIDE_BORDER = DamageTypeImpl.get("minecraft:outside_border");
|
||||||
|
|
||||||
|
DamageType FLY_INTO_WALL = DamageTypeImpl.get("minecraft:fly_into_wall");
|
||||||
|
|
||||||
|
DamageType LIGHTNING_BOLT = DamageTypeImpl.get("minecraft:lightning_bolt");
|
||||||
|
|
||||||
|
DamageType PLAYER_ATTACK = DamageTypeImpl.get("minecraft:player_attack");
|
||||||
|
|
||||||
|
DamageType FREEZE = DamageTypeImpl.get("minecraft:freeze");
|
||||||
|
|
||||||
|
DamageType FALLING_ANVIL = DamageTypeImpl.get("minecraft:falling_anvil");
|
||||||
|
|
||||||
|
DamageType OUT_OF_WORLD = DamageTypeImpl.get("minecraft:out_of_world");
|
||||||
|
|
||||||
|
DamageType MAGIC = DamageTypeImpl.get("minecraft:magic");
|
||||||
|
|
||||||
|
DamageType SWEET_BERRY_BUSH = DamageTypeImpl.get("minecraft:sweet_berry_bush");
|
||||||
|
|
||||||
|
DamageType FIREWORKS = DamageTypeImpl.get("minecraft:fireworks");
|
||||||
|
|
||||||
|
DamageType EXPLOSION = DamageTypeImpl.get("minecraft:explosion");
|
||||||
|
|
||||||
|
DamageType BAD_RESPAWN_POINT = DamageTypeImpl.get("minecraft:bad_respawn_point");
|
||||||
|
|
||||||
|
DamageType STALAGMITE = DamageTypeImpl.get("minecraft:stalagmite");
|
||||||
|
|
||||||
|
DamageType THORNS = DamageTypeImpl.get("minecraft:thorns");
|
||||||
|
|
||||||
|
DamageType INDIRECT_MAGIC = DamageTypeImpl.get("minecraft:indirect_magic");
|
||||||
|
|
||||||
|
DamageType CRAMMING = DamageTypeImpl.get("minecraft:cramming");
|
||||||
|
|
||||||
|
DamageType CACTUS = DamageTypeImpl.get("minecraft:cactus");
|
||||||
|
|
||||||
|
DamageType GENERIC = DamageTypeImpl.get("minecraft:generic");
|
||||||
|
}
|
@ -6,6 +6,7 @@ import net.minestom.server.attribute.AttributeInstance;
|
|||||||
import net.minestom.server.collision.BoundingBox;
|
import net.minestom.server.collision.BoundingBox;
|
||||||
import net.minestom.server.coordinate.Point;
|
import net.minestom.server.coordinate.Point;
|
||||||
import net.minestom.server.coordinate.Vec;
|
import net.minestom.server.coordinate.Vec;
|
||||||
|
import net.minestom.server.entity.damage.Damage;
|
||||||
import net.minestom.server.entity.damage.DamageType;
|
import net.minestom.server.entity.damage.DamageType;
|
||||||
import net.minestom.server.entity.metadata.LivingEntityMeta;
|
import net.minestom.server.entity.metadata.LivingEntityMeta;
|
||||||
import net.minestom.server.event.EventDispatcher;
|
import net.minestom.server.event.EventDispatcher;
|
||||||
@ -48,7 +49,7 @@ public class LivingEntity extends Entity implements EquipmentHandler {
|
|||||||
|
|
||||||
protected boolean isDead;
|
protected boolean isDead;
|
||||||
|
|
||||||
protected DamageType lastDamageSource;
|
protected Damage lastDamage;
|
||||||
|
|
||||||
// Bounding box used for items' pickup (see LivingEntity#setBoundingBox)
|
// Bounding box used for items' pickup (see LivingEntity#setBoundingBox)
|
||||||
protected BoundingBox expandedBoundingBox;
|
protected BoundingBox expandedBoundingBox;
|
||||||
@ -310,26 +311,29 @@ public class LivingEntity extends Entity implements EquipmentHandler {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean damage(@NotNull DamageType type, float amount) {
|
||||||
|
return damage(new Damage(type, null, null, null, amount));
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Damages the entity by a value, the type of the damage also has to be specified.
|
* Damages the entity by a value, the type of the damage also has to be specified.
|
||||||
*
|
*
|
||||||
* @param type the damage type
|
* @param damage the damage to be applied
|
||||||
* @param value the amount of damage
|
|
||||||
* @return true if damage has been applied, false if it didn't
|
* @return true if damage has been applied, false if it didn't
|
||||||
*/
|
*/
|
||||||
public boolean damage(@NotNull DamageType type, float value) {
|
public boolean damage(@NotNull Damage damage) {
|
||||||
if (isDead())
|
if (isDead())
|
||||||
return false;
|
return false;
|
||||||
if (isInvulnerable() || isImmune(type)) {
|
if (isInvulnerable() || isImmune(damage.getType())) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
EntityDamageEvent entityDamageEvent = new EntityDamageEvent(this, type, value, type.getSound(this));
|
EntityDamageEvent entityDamageEvent = new EntityDamageEvent(this, damage, damage.getSound(this));
|
||||||
EventDispatcher.callCancellable(entityDamageEvent, () -> {
|
EventDispatcher.callCancellable(entityDamageEvent, () -> {
|
||||||
// Set the last damage type since the event is not cancelled
|
// Set the last damage type since the event is not cancelled
|
||||||
this.lastDamageSource = entityDamageEvent.getDamageType();
|
this.lastDamage = entityDamageEvent.getDamage();
|
||||||
|
|
||||||
float remainingDamage = entityDamageEvent.getDamage();
|
float remainingDamage = entityDamageEvent.getDamage().getAmount();
|
||||||
|
|
||||||
if (entityDamageEvent.shouldAnimate()) {
|
if (entityDamageEvent.shouldAnimate()) {
|
||||||
sendPacketToViewersAndSelf(new EntityAnimationPacket(getEntityId(), EntityAnimationPacket.Animation.TAKE_DAMAGE));
|
sendPacketToViewersAndSelf(new EntityAnimationPacket(getEntityId(), EntityAnimationPacket.Animation.TAKE_DAMAGE));
|
||||||
@ -410,8 +414,8 @@ public class LivingEntity extends Entity implements EquipmentHandler {
|
|||||||
*
|
*
|
||||||
* @return the last damage source, null if not any
|
* @return the last damage source, null if not any
|
||||||
*/
|
*/
|
||||||
public @Nullable DamageType getLastDamageSource() {
|
public @Nullable Damage getLastDamageSource() {
|
||||||
return lastDamageSource;
|
return lastDamage;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -68,7 +68,6 @@ import net.minestom.server.network.player.PlayerConnection;
|
|||||||
import net.minestom.server.network.player.PlayerSocketConnection;
|
import net.minestom.server.network.player.PlayerSocketConnection;
|
||||||
import net.minestom.server.recipe.Recipe;
|
import net.minestom.server.recipe.Recipe;
|
||||||
import net.minestom.server.recipe.RecipeManager;
|
import net.minestom.server.recipe.RecipeManager;
|
||||||
import net.minestom.server.registry.Registry;
|
|
||||||
import net.minestom.server.resourcepack.ResourcePack;
|
import net.minestom.server.resourcepack.ResourcePack;
|
||||||
import net.minestom.server.scoreboard.BelowNameTag;
|
import net.minestom.server.scoreboard.BelowNameTag;
|
||||||
import net.minestom.server.scoreboard.Team;
|
import net.minestom.server.scoreboard.Team;
|
||||||
@ -97,8 +96,6 @@ import org.jctools.queues.MpscUnboundedXaddArrayQueue;
|
|||||||
import org.jetbrains.annotations.ApiStatus;
|
import org.jetbrains.annotations.ApiStatus;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
import org.jetbrains.annotations.Nullable;
|
import org.jetbrains.annotations.Nullable;
|
||||||
import org.jglrxavpok.hephaistos.nbt.NBT;
|
|
||||||
import org.jglrxavpok.hephaistos.nbt.NBTType;
|
|
||||||
|
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.nio.charset.StandardCharsets;
|
||||||
import java.time.Duration;
|
import java.time.Duration;
|
||||||
@ -416,8 +413,8 @@ public class Player extends LivingEntity implements CommandSender, Localizable,
|
|||||||
|
|
||||||
// get death screen text to the killed player
|
// get death screen text to the killed player
|
||||||
{
|
{
|
||||||
if (lastDamageSource != null) {
|
if (lastDamage != null) {
|
||||||
deathText = lastDamageSource.buildDeathScreenText(this);
|
deathText = lastDamage.buildDeathScreenText(this);
|
||||||
} else { // may happen if killed by the server without applying damage
|
} else { // may happen if killed by the server without applying damage
|
||||||
deathText = Component.text("Killed by poor programming.");
|
deathText = Component.text("Killed by poor programming.");
|
||||||
}
|
}
|
||||||
@ -425,8 +422,8 @@ public class Player extends LivingEntity implements CommandSender, Localizable,
|
|||||||
|
|
||||||
// get death message to chat
|
// get death message to chat
|
||||||
{
|
{
|
||||||
if (lastDamageSource != null) {
|
if (lastDamage != null) {
|
||||||
chatMessage = lastDamageSource.buildDeathMessage(this);
|
chatMessage = lastDamage.buildDeathMessage(this);
|
||||||
} else { // may happen if killed by the server without applying damage
|
} else { // may happen if killed by the server without applying damage
|
||||||
chatMessage = Component.text(getUsername() + " was killed by poor programming.");
|
chatMessage = Component.text(getUsername() + " was killed by poor programming.");
|
||||||
}
|
}
|
||||||
@ -915,7 +912,7 @@ public class Player extends LivingEntity implements CommandSender, Localizable,
|
|||||||
@Override
|
@Override
|
||||||
public boolean isImmune(@NotNull DamageType type) {
|
public boolean isImmune(@NotNull DamageType type) {
|
||||||
if (!getGameMode().canTakeDamage()) {
|
if (!getGameMode().canTakeDamage()) {
|
||||||
return type != DamageType.VOID;
|
return type != DamageType.OUT_OF_WORLD;
|
||||||
}
|
}
|
||||||
return super.isImmune(type);
|
return super.isImmune(type);
|
||||||
}
|
}
|
||||||
|
@ -3,6 +3,7 @@ package net.minestom.server.entity.ai.target;
|
|||||||
import net.minestom.server.entity.Entity;
|
import net.minestom.server.entity.Entity;
|
||||||
import net.minestom.server.entity.EntityCreature;
|
import net.minestom.server.entity.EntityCreature;
|
||||||
import net.minestom.server.entity.ai.TargetSelector;
|
import net.minestom.server.entity.ai.TargetSelector;
|
||||||
|
import net.minestom.server.entity.damage.Damage;
|
||||||
import net.minestom.server.entity.damage.DamageType;
|
import net.minestom.server.entity.damage.DamageType;
|
||||||
import net.minestom.server.entity.damage.EntityDamage;
|
import net.minestom.server.entity.damage.EntityDamage;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
@ -21,8 +22,8 @@ public class LastEntityDamagerTarget extends TargetSelector {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Entity findTarget() {
|
public Entity findTarget() {
|
||||||
final DamageType damageType = entityCreature.getLastDamageSource();
|
final Damage damage = entityCreature.getLastDamageSource();
|
||||||
if (!(damageType instanceof EntityDamage entityDamage)) {
|
if (!(damage instanceof EntityDamage entityDamage)) {
|
||||||
// No damager recorded, return null
|
// No damager recorded, return null
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
179
src/main/java/net/minestom/server/entity/damage/Damage.java
Normal file
179
src/main/java/net/minestom/server/entity/damage/Damage.java
Normal file
@ -0,0 +1,179 @@
|
|||||||
|
package net.minestom.server.entity.damage;
|
||||||
|
|
||||||
|
import net.kyori.adventure.text.Component;
|
||||||
|
import net.minestom.server.coordinate.Point;
|
||||||
|
import net.minestom.server.entity.Entity;
|
||||||
|
import net.minestom.server.entity.LivingEntity;
|
||||||
|
import net.minestom.server.entity.Player;
|
||||||
|
import net.minestom.server.sound.SoundEvent;
|
||||||
|
import net.minestom.server.tag.TagHandler;
|
||||||
|
import net.minestom.server.tag.Taggable;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
import org.jetbrains.annotations.Nullable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents a type of damage, required when calling {@link LivingEntity#damage(Damage)}
|
||||||
|
* and retrieved in {@link net.minestom.server.event.entity.EntityDamageEvent}.
|
||||||
|
* <p>
|
||||||
|
* This class can be extended if you need to include custom fields and/or methods.
|
||||||
|
*/
|
||||||
|
public class Damage implements Taggable {
|
||||||
|
|
||||||
|
private final DamageType type;
|
||||||
|
private final Entity source;
|
||||||
|
private final Entity attacker;
|
||||||
|
private final Point sourcePosition;
|
||||||
|
private final TagHandler tagHandler = TagHandler.newHandler();
|
||||||
|
|
||||||
|
private float amount;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new damage type.
|
||||||
|
*
|
||||||
|
* @param type the type of this damage
|
||||||
|
* @param amount amount of damage
|
||||||
|
*/
|
||||||
|
public Damage(@NotNull DamageType type, @Nullable Entity source, @Nullable Entity attacker, @Nullable Point sourcePosition, float amount) {
|
||||||
|
this.type = type;
|
||||||
|
this.source = source;
|
||||||
|
this.attacker = attacker;
|
||||||
|
this.sourcePosition = sourcePosition;
|
||||||
|
this.amount = amount;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the type of this damage.
|
||||||
|
* <p>
|
||||||
|
* It does not have to be unique to this object.o
|
||||||
|
*
|
||||||
|
* @return the damage type
|
||||||
|
*/
|
||||||
|
public @NotNull DamageType getType() {
|
||||||
|
return type;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the "attacker" of the damage.
|
||||||
|
* This is the indirect cause of the damage, like the shooter of a projectile, or null if there was none.
|
||||||
|
*
|
||||||
|
* @return the attacker
|
||||||
|
*/
|
||||||
|
public @Nullable Entity getAttacker() {
|
||||||
|
return attacker;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the direct source of the damage.
|
||||||
|
* This is the entity that directly causes the damage, like a projectile, or null if there was none.
|
||||||
|
*
|
||||||
|
* @return the source
|
||||||
|
*/
|
||||||
|
public @Nullable Entity getSource() {
|
||||||
|
return source;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the position of the source of the damage, or null if there is none.
|
||||||
|
* This may differ from the source entity's position.
|
||||||
|
*
|
||||||
|
* @return The source position
|
||||||
|
*/
|
||||||
|
public @Nullable Point getSourcePosition() {
|
||||||
|
return sourcePosition;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Builds the death message linked to this damage type.
|
||||||
|
* <p>
|
||||||
|
* Used in {@link Player#kill()} to broadcast the proper message.
|
||||||
|
*
|
||||||
|
* @param killed the player who has been killed
|
||||||
|
* @return the death message, null to do not send anything
|
||||||
|
*/
|
||||||
|
public @Nullable Component buildDeathMessage(@NotNull Player killed) {
|
||||||
|
return Component.translatable("death.attack." + type.messageId(), Component.text(killed.getUsername()));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convenient method to create an {@link EntityProjectileDamage}.
|
||||||
|
*
|
||||||
|
* @param shooter the shooter
|
||||||
|
* @param projectile the actual projectile
|
||||||
|
* @param amount amount of damage
|
||||||
|
* @return a new {@link EntityProjectileDamage}
|
||||||
|
*/
|
||||||
|
public static @NotNull Damage fromProjectile(@Nullable Entity shooter, @NotNull Entity projectile, float amount) {
|
||||||
|
return new EntityProjectileDamage(shooter, projectile, amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convenient method to create an {@link EntityDamage}.
|
||||||
|
*
|
||||||
|
* @param player the player damager
|
||||||
|
* @param amount amount of damage
|
||||||
|
* @return a new {@link EntityDamage}
|
||||||
|
*/
|
||||||
|
public static @NotNull EntityDamage fromPlayer(@NotNull Player player, float amount) {
|
||||||
|
return new EntityDamage(player, amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convenient method to create an {@link EntityDamage}.
|
||||||
|
*
|
||||||
|
* @param entity the entity damager
|
||||||
|
* @param amount amount of damage
|
||||||
|
* @return a new {@link EntityDamage}
|
||||||
|
*/
|
||||||
|
public static @NotNull EntityDamage fromEntity(@NotNull Entity entity, float amount) {
|
||||||
|
return new EntityDamage(entity, amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static @NotNull PositionalDamage fromPosition(@NotNull DamageType type, @NotNull Point sourcePosition, float amount) {
|
||||||
|
return new PositionalDamage(type, sourcePosition, amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Builds the text sent to a player in his death screen.
|
||||||
|
*
|
||||||
|
* @param killed the player who has been killed
|
||||||
|
* @return the death screen text, null to do not send anything
|
||||||
|
*/
|
||||||
|
public @Nullable Component buildDeathScreenText(@NotNull Player killed) {
|
||||||
|
return Component.translatable("death.attack." + type.messageId());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sound event to play when the given entity is hit by this damage. Possible to return null if no sound should be played
|
||||||
|
*
|
||||||
|
* @param entity the entity hit by this damage
|
||||||
|
* @return the sound to play when the given entity is hurt by this damage type. Can be null if no sound should play
|
||||||
|
*/
|
||||||
|
public @Nullable SoundEvent getSound(@NotNull LivingEntity entity) {
|
||||||
|
if (entity instanceof Player) {
|
||||||
|
return getPlayerSound((Player) entity);
|
||||||
|
}
|
||||||
|
return getGenericSound(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected SoundEvent getGenericSound(@NotNull LivingEntity entity) {
|
||||||
|
return SoundEvent.ENTITY_GENERIC_HURT;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected SoundEvent getPlayerSound(@NotNull Player player) {
|
||||||
|
if (type == DamageType.ON_FIRE) return SoundEvent.ENTITY_PLAYER_HURT_ON_FIRE;
|
||||||
|
return SoundEvent.ENTITY_PLAYER_HURT;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public @NotNull TagHandler tagHandler() {
|
||||||
|
return tagHandler;
|
||||||
|
}
|
||||||
|
|
||||||
|
public float getAmount() {
|
||||||
|
return amount;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setAmount(float amount) {
|
||||||
|
this.amount = amount;
|
||||||
|
}
|
||||||
|
}
|
@ -1,131 +1,60 @@
|
|||||||
package net.minestom.server.entity.damage;
|
package net.minestom.server.entity.damage;
|
||||||
|
|
||||||
import net.kyori.adventure.text.Component;
|
import net.minestom.server.registry.ProtocolObject;
|
||||||
import net.minestom.server.entity.Entity;
|
import net.minestom.server.registry.Registry;
|
||||||
import net.minestom.server.entity.LivingEntity;
|
import net.minestom.server.utils.NamespaceID;
|
||||||
import net.minestom.server.entity.Player;
|
import org.jetbrains.annotations.Contract;
|
||||||
import net.minestom.server.sound.SoundEvent;
|
|
||||||
import net.minestom.server.tag.TagHandler;
|
|
||||||
import net.minestom.server.tag.Taggable;
|
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
import org.jetbrains.annotations.Nullable;
|
import org.jetbrains.annotations.Nullable;
|
||||||
|
import org.jglrxavpok.hephaistos.nbt.NBTCompound;
|
||||||
|
|
||||||
/**
|
import java.util.Collection;
|
||||||
* Represents a type of damage, required when calling {@link LivingEntity#damage(DamageType, float)}
|
|
||||||
* and retrieved in {@link net.minestom.server.event.entity.EntityDamageEvent}.
|
|
||||||
* <p>
|
|
||||||
* This class can be extended if you need to include custom fields and/or methods.
|
|
||||||
*/
|
|
||||||
public class DamageType implements Taggable {
|
|
||||||
|
|
||||||
public static final DamageType VOID = new DamageType("attack.outOfWorld");
|
|
||||||
public static final DamageType GRAVITY = new DamageType("attack.fall");
|
|
||||||
public static final DamageType ON_FIRE = new DamageType("attack.onFire") {
|
|
||||||
@Override
|
|
||||||
protected SoundEvent getPlayerSound(@NotNull Player player) {
|
|
||||||
return SoundEvent.ENTITY_PLAYER_HURT_ON_FIRE;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
private final String identifier;
|
|
||||||
private final TagHandler tagHandler = TagHandler.newHandler();
|
|
||||||
|
|
||||||
|
public sealed interface DamageType extends ProtocolObject, DamageTypes permits DamageTypeImpl {
|
||||||
/**
|
/**
|
||||||
* Creates a new damage type.
|
* Returns the damage type registry.
|
||||||
*
|
*
|
||||||
* @param identifier the identifier of this damage type,
|
* @return the damage type registry
|
||||||
* does not need to be unique
|
|
||||||
*/
|
*/
|
||||||
public DamageType(@NotNull String identifier) {
|
@Contract(pure = true)
|
||||||
this.identifier = identifier;
|
@NotNull Registry.DamageTypeEntry registry();
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the identifier of this damage type.
|
|
||||||
* <p>
|
|
||||||
* It does not have to be unique to this object.o
|
|
||||||
*
|
|
||||||
* @return the damage type identifier
|
|
||||||
*/
|
|
||||||
public @NotNull String getIdentifier() {
|
|
||||||
return identifier;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Builds the death message linked to this damage type.
|
|
||||||
* <p>
|
|
||||||
* Used in {@link Player#kill()} to broadcast the proper message.
|
|
||||||
*
|
|
||||||
* @param killed the player who has been killed
|
|
||||||
* @return the death message, null to do not send anything
|
|
||||||
*/
|
|
||||||
public @Nullable Component buildDeathMessage(@NotNull Player killed) {
|
|
||||||
return Component.translatable("death." + identifier, Component.text(killed.getUsername()));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convenient method to create an {@link EntityProjectileDamage}.
|
|
||||||
*
|
|
||||||
* @param shooter the shooter
|
|
||||||
* @param projectile the actual projectile
|
|
||||||
* @return a new {@link EntityProjectileDamage}
|
|
||||||
*/
|
|
||||||
public static @NotNull DamageType fromProjectile(@Nullable Entity shooter, @NotNull Entity projectile) {
|
|
||||||
return new EntityProjectileDamage(shooter, projectile);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convenient method to create an {@link EntityDamage}.
|
|
||||||
*
|
|
||||||
* @param player the player damager
|
|
||||||
* @return a new {@link EntityDamage}
|
|
||||||
*/
|
|
||||||
public static @NotNull EntityDamage fromPlayer(@NotNull Player player) {
|
|
||||||
return new EntityDamage(player);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convenient method to create an {@link EntityDamage}.
|
|
||||||
*
|
|
||||||
* @param entity the entity damager
|
|
||||||
* @return a new {@link EntityDamage}
|
|
||||||
*/
|
|
||||||
public static @NotNull EntityDamage fromEntity(@NotNull Entity entity) {
|
|
||||||
return new EntityDamage(entity);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Builds the text sent to a player in his death screen.
|
|
||||||
*
|
|
||||||
* @param killed the player who has been killed
|
|
||||||
* @return the death screen text, null to do not send anything
|
|
||||||
*/
|
|
||||||
public @Nullable Component buildDeathScreenText(@NotNull Player killed) {
|
|
||||||
return Component.translatable("death." + identifier);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sound event to play when the given entity is hit by this damage. Possible to return null if no sound should be played
|
|
||||||
*
|
|
||||||
* @param entity the entity hit by this damage
|
|
||||||
* @return the sound to play when the given entity is hurt by this damage type. Can be null if no sound should play
|
|
||||||
*/
|
|
||||||
public @Nullable SoundEvent getSound(@NotNull LivingEntity entity) {
|
|
||||||
if (entity instanceof Player) {
|
|
||||||
return getPlayerSound((Player) entity);
|
|
||||||
}
|
|
||||||
return getGenericSound(entity);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected SoundEvent getGenericSound(@NotNull LivingEntity entity) {
|
|
||||||
return SoundEvent.ENTITY_GENERIC_HURT;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected SoundEvent getPlayerSound(@NotNull Player player) {
|
|
||||||
return SoundEvent.ENTITY_PLAYER_HURT;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public @NotNull TagHandler tagHandler() {
|
default @NotNull NamespaceID namespace() {
|
||||||
return tagHandler;
|
return registry().namespace();
|
||||||
|
}
|
||||||
|
|
||||||
|
default double exhaustion() {
|
||||||
|
return registry().exhaustion();
|
||||||
|
}
|
||||||
|
|
||||||
|
default String messageId() {
|
||||||
|
return registry().messageId();
|
||||||
|
}
|
||||||
|
|
||||||
|
default String scaling() {
|
||||||
|
return registry().scaling();
|
||||||
|
}
|
||||||
|
|
||||||
|
NBTCompound asNBT();
|
||||||
|
|
||||||
|
static @NotNull Collection<@NotNull DamageType> values() {
|
||||||
|
return DamageTypeImpl.values();
|
||||||
|
}
|
||||||
|
|
||||||
|
static DamageType fromNamespaceId(@NotNull String namespaceID) {
|
||||||
|
return DamageTypeImpl.getSafe(namespaceID);
|
||||||
|
}
|
||||||
|
|
||||||
|
static DamageType fromNamespaceId(@NotNull NamespaceID namespaceID) {
|
||||||
|
return fromNamespaceId(namespaceID.asString());
|
||||||
|
}
|
||||||
|
|
||||||
|
static @Nullable DamageType fromId(int id) {
|
||||||
|
return DamageTypeImpl.getId(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
static NBTCompound getNBT() {
|
||||||
|
return DamageTypeImpl.getNBT();
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -0,0 +1,77 @@
|
|||||||
|
package net.minestom.server.entity.damage;
|
||||||
|
|
||||||
|
import net.minestom.server.registry.Registry;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
import org.jglrxavpok.hephaistos.nbt.NBT;
|
||||||
|
import org.jglrxavpok.hephaistos.nbt.NBTCompound;
|
||||||
|
import org.jglrxavpok.hephaistos.nbt.NBTType;
|
||||||
|
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
|
record DamageTypeImpl(Registry.DamageTypeEntry registry, int id) implements DamageType {
|
||||||
|
private static final Registry.Container<DamageType> CONTAINER;
|
||||||
|
|
||||||
|
static {
|
||||||
|
AtomicInteger i = new AtomicInteger();
|
||||||
|
CONTAINER = Registry.createContainer(Registry.Resource.DAMAGE_TYPES,
|
||||||
|
(namespace, properties) -> new DamageTypeImpl(Registry.damageType(namespace, properties), i.getAndIncrement()));
|
||||||
|
}
|
||||||
|
|
||||||
|
static DamageType get(@NotNull String namespace) {
|
||||||
|
return CONTAINER.get(namespace);
|
||||||
|
}
|
||||||
|
|
||||||
|
static DamageType getSafe(@NotNull String namespace) {
|
||||||
|
return CONTAINER.getSafe(namespace);
|
||||||
|
}
|
||||||
|
|
||||||
|
static DamageType getId(int id) {
|
||||||
|
return CONTAINER.getId(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public NBTCompound asNBT() {
|
||||||
|
var elem = new HashMap<String, NBT>();
|
||||||
|
elem.put("exhaustion", NBT.Float(registry.exhaustion()));
|
||||||
|
elem.put("message_id", NBT.String(registry.messageId()));
|
||||||
|
elem.put("scaling", NBT.String(registry.scaling()));
|
||||||
|
return NBT.Compound(elem);
|
||||||
|
}
|
||||||
|
|
||||||
|
static Collection<DamageType> values() {
|
||||||
|
return CONTAINER.values();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return name();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int id() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static NBTCompound lazyNbt = null;
|
||||||
|
|
||||||
|
static NBTCompound getNBT() {
|
||||||
|
if (lazyNbt == null) {
|
||||||
|
var damageTypes = values().stream()
|
||||||
|
.map((damageType) -> NBT.Compound(Map.of(
|
||||||
|
"id", NBT.Int(damageType.id()),
|
||||||
|
"name", NBT.String(damageType.name()),
|
||||||
|
"element", damageType.asNBT()
|
||||||
|
)))
|
||||||
|
.toList();
|
||||||
|
|
||||||
|
lazyNbt = NBT.Compound(Map.of(
|
||||||
|
"type", NBT.String("minecraft:damage_type"),
|
||||||
|
"value", NBT.List(NBTType.TAG_Compound, damageTypes)
|
||||||
|
));
|
||||||
|
}
|
||||||
|
return lazyNbt;
|
||||||
|
}
|
||||||
|
}
|
@ -6,13 +6,10 @@ import org.jetbrains.annotations.NotNull;
|
|||||||
/**
|
/**
|
||||||
* Represents damage inflicted by an {@link Entity}.
|
* Represents damage inflicted by an {@link Entity}.
|
||||||
*/
|
*/
|
||||||
public class EntityDamage extends DamageType {
|
public class EntityDamage extends Damage {
|
||||||
|
|
||||||
private final Entity source;
|
public EntityDamage(@NotNull Entity source, float amount) {
|
||||||
|
super(DamageType.MOB_ATTACK, source, source, null, amount);
|
||||||
public EntityDamage(@NotNull Entity source) {
|
|
||||||
super("entity_source");
|
|
||||||
this.source = source;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -20,8 +17,13 @@ public class EntityDamage extends DamageType {
|
|||||||
*
|
*
|
||||||
* @return the source
|
* @return the source
|
||||||
*/
|
*/
|
||||||
@NotNull
|
@Override
|
||||||
public Entity getSource() {
|
public @NotNull Entity getSource() {
|
||||||
return source;
|
return super.getSource();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public @NotNull Entity getAttacker() {
|
||||||
|
return getSource();
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -7,15 +7,10 @@ import org.jetbrains.annotations.Nullable;
|
|||||||
/**
|
/**
|
||||||
* Represents damage inflicted by an entity, via a projectile.
|
* Represents damage inflicted by an entity, via a projectile.
|
||||||
*/
|
*/
|
||||||
public class EntityProjectileDamage extends DamageType {
|
public class EntityProjectileDamage extends Damage {
|
||||||
|
|
||||||
private final Entity shooter;
|
public EntityProjectileDamage(@Nullable Entity shooter, @NotNull Entity projectile, float amount) {
|
||||||
private final Entity projectile;
|
super(DamageType.MOB_PROJECTILE, projectile, shooter, null, amount);
|
||||||
|
|
||||||
public EntityProjectileDamage(@Nullable Entity shooter, @NotNull Entity projectile) {
|
|
||||||
super("projectile_source");
|
|
||||||
this.shooter = shooter;
|
|
||||||
this.projectile = projectile;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -25,7 +20,7 @@ public class EntityProjectileDamage extends DamageType {
|
|||||||
*/
|
*/
|
||||||
@NotNull
|
@NotNull
|
||||||
public Entity getProjectile() {
|
public Entity getProjectile() {
|
||||||
return projectile;
|
return getSource();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -35,6 +30,11 @@ public class EntityProjectileDamage extends DamageType {
|
|||||||
*/
|
*/
|
||||||
@Nullable
|
@Nullable
|
||||||
public Entity getShooter() {
|
public Entity getShooter() {
|
||||||
return shooter;
|
return getAttacker();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public @NotNull Entity getSource() {
|
||||||
|
return super.getSource();
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -0,0 +1,19 @@
|
|||||||
|
package net.minestom.server.entity.damage;
|
||||||
|
|
||||||
|
import net.minestom.server.coordinate.Point;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents damage that is associated with a certain position.
|
||||||
|
*/
|
||||||
|
public class PositionalDamage extends Damage {
|
||||||
|
|
||||||
|
public PositionalDamage(@NotNull DamageType type, @NotNull Point sourcePosition, float amount) {
|
||||||
|
super(type, null, null, sourcePosition, amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public @NotNull Point getSourcePosition() {
|
||||||
|
return super.getSourcePosition();
|
||||||
|
}
|
||||||
|
}
|
@ -2,6 +2,7 @@ package net.minestom.server.event.entity;
|
|||||||
|
|
||||||
import net.minestom.server.entity.Entity;
|
import net.minestom.server.entity.Entity;
|
||||||
import net.minestom.server.entity.LivingEntity;
|
import net.minestom.server.entity.LivingEntity;
|
||||||
|
import net.minestom.server.entity.damage.Damage;
|
||||||
import net.minestom.server.entity.damage.DamageType;
|
import net.minestom.server.entity.damage.DamageType;
|
||||||
import net.minestom.server.event.trait.CancellableEvent;
|
import net.minestom.server.event.trait.CancellableEvent;
|
||||||
import net.minestom.server.event.trait.EntityInstanceEvent;
|
import net.minestom.server.event.trait.EntityInstanceEvent;
|
||||||
@ -15,17 +16,14 @@ import org.jetbrains.annotations.Nullable;
|
|||||||
public class EntityDamageEvent implements EntityInstanceEvent, CancellableEvent {
|
public class EntityDamageEvent implements EntityInstanceEvent, CancellableEvent {
|
||||||
|
|
||||||
private final Entity entity;
|
private final Entity entity;
|
||||||
private final DamageType damageType;
|
private final Damage damage;
|
||||||
private float damage;
|
|
||||||
private SoundEvent sound;
|
private SoundEvent sound;
|
||||||
private boolean animation = true;
|
private boolean animation = true;
|
||||||
|
|
||||||
private boolean cancelled;
|
private boolean cancelled;
|
||||||
|
|
||||||
public EntityDamageEvent(@NotNull LivingEntity entity, @NotNull DamageType damageType,
|
public EntityDamageEvent(@NotNull LivingEntity entity, @NotNull Damage damage, @Nullable SoundEvent sound) {
|
||||||
float damage, @Nullable SoundEvent sound) {
|
|
||||||
this.entity = entity;
|
this.entity = entity;
|
||||||
this.damageType = damageType;
|
|
||||||
this.damage = damage;
|
this.damage = damage;
|
||||||
this.sound = sound;
|
this.sound = sound;
|
||||||
}
|
}
|
||||||
@ -42,28 +40,10 @@ public class EntityDamageEvent implements EntityInstanceEvent, CancellableEvent
|
|||||||
* @return the damage type
|
* @return the damage type
|
||||||
*/
|
*/
|
||||||
@NotNull
|
@NotNull
|
||||||
public DamageType getDamageType() {
|
public Damage getDamage() {
|
||||||
return damageType;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the damage amount.
|
|
||||||
*
|
|
||||||
* @return the damage amount
|
|
||||||
*/
|
|
||||||
public float getDamage() {
|
|
||||||
return damage;
|
return damage;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Changes the damage amount.
|
|
||||||
*
|
|
||||||
* @param damage the new damage amount
|
|
||||||
*/
|
|
||||||
public void setDamage(float damage) {
|
|
||||||
this.damage = damage;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the damage sound.
|
* Gets the damage sound.
|
||||||
*
|
*
|
||||||
|
@ -53,6 +53,11 @@ public final class Registry {
|
|||||||
return new PotionEffectEntry(namespace, main, null);
|
return new PotionEffectEntry(namespace, main, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ApiStatus.Internal
|
||||||
|
public static DamageTypeEntry damageType(String namespace, @NotNull Properties main) {
|
||||||
|
return new DamageTypeEntry(namespace, main, null);
|
||||||
|
}
|
||||||
|
|
||||||
@ApiStatus.Internal
|
@ApiStatus.Internal
|
||||||
public static Map<String, Map<String, Object>> load(Resource resource) {
|
public static Map<String, Map<String, Object>> load(Resource resource) {
|
||||||
Map<String, Map<String, Object>> map = new HashMap<>();
|
Map<String, Map<String, Object>> map = new HashMap<>();
|
||||||
@ -409,6 +414,23 @@ public final class Registry {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public record DamageTypeEntry(NamespaceID namespace, float exhaustion,
|
||||||
|
String messageId,
|
||||||
|
String scaling,
|
||||||
|
@Nullable String effects,
|
||||||
|
@Nullable String deathMessageType,
|
||||||
|
Properties custom) implements Entry {
|
||||||
|
public DamageTypeEntry(String namespace, Properties main, Properties custom) {
|
||||||
|
this(NamespaceID.from(namespace),
|
||||||
|
(float) main.getDouble("exhaustion"),
|
||||||
|
main.getString("message_id"),
|
||||||
|
main.getString("scaling"),
|
||||||
|
main.getString("effects"),
|
||||||
|
main.getString("death_message_type"),
|
||||||
|
custom);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public record EnchantmentEntry(NamespaceID namespace, int id,
|
public record EnchantmentEntry(NamespaceID namespace, int id,
|
||||||
String translationKey,
|
String translationKey,
|
||||||
double maxLevel,
|
double maxLevel,
|
||||||
|
Loading…
Reference in New Issue
Block a user