diff --git a/src/main/java/net/minestom/server/tag/Tag.java b/src/main/java/net/minestom/server/tag/Tag.java index c60dfbf73..6a31f845f 100644 --- a/src/main/java/net/minestom/server/tag/Tag.java +++ b/src/main/java/net/minestom/server/tag/Tag.java @@ -1,6 +1,7 @@ package net.minestom.server.tag; import org.jetbrains.annotations.ApiStatus; +import org.jetbrains.annotations.Contract; import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.Nullable; import org.jglrxavpok.hephaistos.nbt.NBT; @@ -23,28 +24,64 @@ public class Tag { private final String key; private final Function readFunction; private final BiConsumer writeConsumer; - private volatile Supplier defaultValue; + + private final Supplier defaultValue; protected Tag(@NotNull String key, - @NotNull Function readFunction, - @NotNull BiConsumer writeConsumer) { + @NotNull Function readFunction, + @NotNull BiConsumer writeConsumer, + @Nullable Supplier defaultValue) { this.key = key; this.readFunction = readFunction; this.writeConsumer = writeConsumer; + this.defaultValue = defaultValue; + } + + protected Tag(@NotNull String key, + @NotNull Function readFunction, + @NotNull BiConsumer writeConsumer) { + this(key, readFunction, writeConsumer, null); } public @NotNull String getKey() { return key; } + @Contract(value = "_ -> new", pure = true) public Tag defaultValue(@NotNull Supplier defaultValue) { - this.defaultValue = defaultValue; - return this; + return new Tag<>(key, readFunction, writeConsumer, defaultValue); } + @Contract(value = "_ -> new", pure = true) public Tag defaultValue(@NotNull T defaultValue) { - defaultValue(() -> defaultValue); - return this; + return new Tag<>(key, readFunction, writeConsumer, () -> defaultValue); + } + + @Contract(value = "_, _ -> new", pure = true) + public Tag map(@NotNull Function readMap, + @NotNull Function writeMap) { + return new Tag(key, + // Read + nbtCompound -> { + final var old = readFunction.apply(nbtCompound); + if (old == null) { + return null; + } + return readMap.apply(old); + }, + // Write + (nbtCompound, r) -> { + var n = writeMap.apply(r); + writeConsumer.accept(nbtCompound, n); + }, + // Default value + () -> { + if (defaultValue == null) { + return null; + } + var old = defaultValue.get(); + return readMap.apply(old); + }); } public @Nullable T read(@NotNull NBTCompound nbtCompound) { @@ -141,7 +178,20 @@ public class Tag { public static @NotNull Tag Custom(@NotNull String key, @NotNull TagSerializer serializer) { return new Tag<>(key, - nbtCompound -> serializer.read(TagReadable.fromCompound(nbtCompound)), - (nbtCompound, value) -> serializer.write(TagWritable.fromCompound(nbtCompound), value)); + nbtCompound -> { + final var compound = nbtCompound.getCompound(key); + if (compound == null) { + return null; + } + return serializer.read(TagReadable.fromCompound(compound)); + }, + (nbtCompound, value) -> { + var compound = nbtCompound.getCompound(key); + if (compound == null) { + compound = new NBTCompound(); + nbtCompound.set(key, compound); + } + serializer.write(TagWritable.fromCompound(compound), value); + }); } }