From b47b3834a4fc5d956394bfc2055be2e88dcdebf1 Mon Sep 17 00:00:00 2001 From: Will FP Date: Mon, 4 Nov 2024 17:23:14 +0000 Subject: [PATCH] Added 1.21.3 support --- build.gradle.kts | 8 +- .../com/willfp/eco/core/Prerequisite.java | 24 ++- .../willfp/eco/core/proxy/ProxyConstants.java | 4 +- .../eco/internal/proxy/EcoProxyFactory.kt | 3 +- eco-core/core-nms/v1_21_3/build.gradle.kts | 50 +++++ .../spigot/proxy/v1_21_3/BukkitCommands.kt | 35 ++++ .../proxy/v1_21_3/CommonsInitializer.kt | 171 ++++++++++++++++++ .../spigot/proxy/v1_21_3/DisplayName.kt | 57 ++++++ .../proxy/v1_21_3/DummyEntityFactory.kt | 15 ++ .../proxy/v1_21_3/EntityControllerFactory.kt | 12 ++ .../ExtendedPersistentDataContainerFactory.kt | 91 ++++++++++ .../proxy/v1_21_3/FastItemStackFactory.kt | 12 ++ .../proxy/v1_21_3/MiniMessageTranslator.kt | 33 ++++ .../spigot/proxy/v1_21_3/PacketHandler.kt | 47 +++++ .../spigot/proxy/v1_21_3/SNBTConverter.kt | 80 ++++++++ .../internal/spigot/proxy/v1_21_3/Skull.kt | 18 ++ .../eco/internal/spigot/proxy/v1_21_3/TPS.kt | 11 ++ .../v1_21_3/entity/EcoEntityController.kt | 95 ++++++++++ .../NewItemsPacketOpenWindowMerchant.kt | 35 ++++ .../packet/NewItemsPacketSetCreativeSlot.kt | 19 ++ .../spigot/recipes/CraftingRecipeListener.kt | 29 ++- gradle.properties | 2 +- settings.gradle.kts | 1 + 23 files changed, 837 insertions(+), 15 deletions(-) create mode 100644 eco-core/core-nms/v1_21_3/build.gradle.kts create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/BukkitCommands.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/CommonsInitializer.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/DisplayName.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/DummyEntityFactory.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/EntityControllerFactory.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/ExtendedPersistentDataContainerFactory.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/FastItemStackFactory.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/MiniMessageTranslator.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/PacketHandler.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/SNBTConverter.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/Skull.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/TPS.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/entity/EcoEntityController.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/packet/NewItemsPacketOpenWindowMerchant.kt create mode 100644 eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/packet/NewItemsPacketSetCreativeSlot.kt diff --git a/build.gradle.kts b/build.gradle.kts index 5bee9a7be..0144d5318 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -10,7 +10,7 @@ buildscript { plugins { id("java-library") - id("io.github.goooler.shadow") version "8.1.7" + id("com.gradleup.shadow") version "8.3.5" id("maven-publish") id("java") kotlin("jvm") version "1.9.21" @@ -32,13 +32,14 @@ dependencies { implementation(project(path = ":eco-core:core-nms:v1_20_R2", configuration = "reobf")) implementation(project(path = ":eco-core:core-nms:v1_20_R3", configuration = "reobf")) implementation(project(path = ":eco-core:core-nms:v1_21", configuration = "reobf")) + implementation(project(path = ":eco-core:core-nms:v1_21_3", configuration = "reobf")) } allprojects { apply(plugin = "java") apply(plugin = "java-library") apply(plugin = "maven-publish") - apply(plugin = "io.github.goooler.shadow") + apply(plugin = "com.gradleup.shadow") apply(plugin = "kotlin") repositories { @@ -48,6 +49,9 @@ allprojects { content { includeGroupByRegex("com\\.github\\..*") } } + // Paper + maven("https://repo.papermc.io/repository/maven-public/") + // SuperiorSkyblock2 maven("https://repo.bg-software.com/repository/api/") diff --git a/eco-api/src/main/java/com/willfp/eco/core/Prerequisite.java b/eco-api/src/main/java/com/willfp/eco/core/Prerequisite.java index 6efa7b97b..919a4a100 100644 --- a/eco-api/src/main/java/com/willfp/eco/core/Prerequisite.java +++ b/eco-api/src/main/java/com/willfp/eco/core/Prerequisite.java @@ -38,15 +38,23 @@ public class Prerequisite { ); /** - * Requires the server to be running 1.21. + * Requires the server to be running at least 1.21.3. + */ + public static final Prerequisite HAS_1_21_3 = new Prerequisite( + () -> ProxyConstants.NMS_VERSION.contains("1_21_3"), + "Requires server to be running 1.21.3+" + ); + + /** + * Requires the server to be running at least 1.21. */ public static final Prerequisite HAS_1_21 = new Prerequisite( - () -> ProxyConstants.NMS_VERSION.contains("1_21"), + () -> ProxyConstants.NMS_VERSION.contains("1_21") || HAS_1_21_3.isMet(), "Requires server to be running 1.21+" ); /** - * Requires the server to be running 1.20.5. + * Requires the server to be running at least 1.20.5. */ public static final Prerequisite HAS_1_20_5 = new Prerequisite( () -> (ProxyConstants.NMS_VERSION.contains("1_20_") && !ProxyConstants.NMS_VERSION.contains("R")) @@ -55,7 +63,7 @@ public class Prerequisite { ); /** - * Requires the server to be running 1.20.3. + * Requires the server to be running at least 1.20.3. */ public static final Prerequisite HAS_1_20_3 = new Prerequisite( () -> ProxyConstants.NMS_VERSION.contains("20_R3") || HAS_1_20_5.isMet(), @@ -63,7 +71,7 @@ public class Prerequisite { ); /** - * Requires the server to be running 1.20. + * Requires the server to be running at least 1.20. */ public static final Prerequisite HAS_1_20 = new Prerequisite( () -> ProxyConstants.NMS_VERSION.contains("20") || HAS_1_20_3.isMet(), @@ -71,7 +79,7 @@ public class Prerequisite { ); /** - * Requires the server to be running 1.19.4. + * Requires the server to be running at least 1.19.4. */ public static final Prerequisite HAS_1_19_4 = new Prerequisite( () -> ProxyConstants.NMS_VERSION.contains("19_R3") || HAS_1_20.isMet(), @@ -79,7 +87,7 @@ public class Prerequisite { ); /** - * Requires the server to be running 1.19. + * Requires the server to be running at least 1.19. */ public static final Prerequisite HAS_1_19 = new Prerequisite( () -> ProxyConstants.NMS_VERSION.contains("19") || HAS_1_20.isMet(), @@ -87,7 +95,7 @@ public class Prerequisite { ); /** - * Requires the server to be running 1.18. + * Requires the server to be running at least 1.18. */ public static final Prerequisite HAS_1_18 = new Prerequisite( () -> ProxyConstants.NMS_VERSION.contains("18") || HAS_1_19.isMet(), diff --git a/eco-api/src/main/java/com/willfp/eco/core/proxy/ProxyConstants.java b/eco-api/src/main/java/com/willfp/eco/core/proxy/ProxyConstants.java index 9bc8c2f95..95ef40488 100644 --- a/eco-api/src/main/java/com/willfp/eco/core/proxy/ProxyConstants.java +++ b/eco-api/src/main/java/com/willfp/eco/core/proxy/ProxyConstants.java @@ -29,7 +29,8 @@ public final class ProxyConstants { "v1_20_R1", "v1_20_R2", "v1_20_R3", - "v1_21" + "v1_21", + "v1_21_3" ); private ProxyConstants() { @@ -39,6 +40,7 @@ private ProxyConstants() { private static String convertVersion(@NotNull final String version) { return switch (version) { case "v1_21_1" -> "v1_21"; + case "v1_21_2" -> "v1_21_3"; default -> version; }; } diff --git a/eco-core/core-backend/src/main/kotlin/com/willfp/eco/internal/proxy/EcoProxyFactory.kt b/eco-core/core-backend/src/main/kotlin/com/willfp/eco/internal/proxy/EcoProxyFactory.kt index ad3f8b495..814aa09ba 100644 --- a/eco-core/core-backend/src/main/kotlin/com/willfp/eco/internal/proxy/EcoProxyFactory.kt +++ b/eco-core/core-backend/src/main/kotlin/com/willfp/eco/internal/proxy/EcoProxyFactory.kt @@ -50,8 +50,7 @@ class EcoProxyFactory( ) } else { ProxyError( - "Could not initialize proxy. If you're seeing this error message" - + ", something has gone badly wrong. This almost definitely isn't user error, blame the developer.", + "Could not initialize proxy. Are you running a supported server version?", e ) } diff --git a/eco-core/core-nms/v1_21_3/build.gradle.kts b/eco-core/core-nms/v1_21_3/build.gradle.kts new file mode 100644 index 000000000..0316fca79 --- /dev/null +++ b/eco-core/core-nms/v1_21_3/build.gradle.kts @@ -0,0 +1,50 @@ +plugins { + id("io.papermc.paperweight.userdev") +} + +group = "com.willfp" +version = rootProject.version + +dependencies { + implementation(project(":eco-core:core-nms:modern")) + implementation(project(":eco-core:core-nms:common")) + paperweight.paperDevBundle("1.21.3-R0.1-SNAPSHOT") + + implementation("net.kyori:adventure-text-minimessage:4.11.0") { + version { + strictly("4.11.0") + } + exclude(group = "net.kyori", module = "adventure-api") + } +} + +tasks { + build { + dependsOn(reobfJar) + } + + reobfJar { + mustRunAfter(shadowJar) + } + + shadowJar { + relocate( + "com.willfp.eco.internal.spigot.proxy.common", + "com.willfp.eco.internal.spigot.proxy.v1_21_3.common" + ) + relocate( + "net.kyori.adventure.text.minimessage", + "com.willfp.eco.internal.spigot.proxy.v1_21_3.minimessage" + ) + } + + compileJava { + options.release = 21 + } + + compileKotlin { + kotlinOptions { + jvmTarget = "21" + } + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/BukkitCommands.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/BukkitCommands.kt new file mode 100644 index 000000000..b4d48e534 --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/BukkitCommands.kt @@ -0,0 +1,35 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3 + +import com.willfp.eco.core.command.PluginCommandBase +import com.willfp.eco.internal.spigot.proxy.BukkitCommandsProxy +import org.bukkit.Bukkit +import org.bukkit.command.Command +import org.bukkit.command.SimpleCommandMap +import org.bukkit.craftbukkit.CraftServer +import java.lang.reflect.Field + +class BukkitCommands : BukkitCommandsProxy { + private val knownCommandsField: Field by lazy { + SimpleCommandMap::class.java.getDeclaredField("knownCommands") + .apply { + isAccessible = true + } + } + + @Suppress("UNCHECKED_CAST") + private val knownCommands: MutableMap + get() = knownCommandsField.get(getCommandMap()) as MutableMap + + override fun getCommandMap(): SimpleCommandMap { + return (Bukkit.getServer() as CraftServer).commandMap + } + + override fun syncCommands() { + (Bukkit.getServer() as CraftServer).syncCommands() + } + + override fun unregisterCommand(command: PluginCommandBase) { + knownCommands.remove(command.name) + knownCommands.remove("${command.plugin.name.lowercase()}:${command.name}") + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/CommonsInitializer.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/CommonsInitializer.kt new file mode 100644 index 000000000..8c4be7519 --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/CommonsInitializer.kt @@ -0,0 +1,171 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3 + +import com.willfp.eco.core.EcoPlugin +import com.willfp.eco.internal.spigot.proxy.CommonsInitializerProxy +import com.willfp.eco.internal.spigot.proxy.common.CommonsProvider +import com.willfp.eco.internal.spigot.proxy.common.packet.PacketInjectorListener +import com.willfp.eco.internal.spigot.proxy.common.toResourceLocation +import net.kyori.adventure.text.Component +import net.kyori.adventure.text.serializer.json.JSONComponentSerializer +import net.minecraft.core.component.DataComponents +import net.minecraft.core.registries.BuiltInRegistries +import net.minecraft.nbt.CompoundTag +import net.minecraft.nbt.Tag +import net.minecraft.resources.ResourceLocation +import net.minecraft.server.level.ServerPlayer +import net.minecraft.world.entity.PathfinderMob +import net.minecraft.world.item.Item +import org.bukkit.Bukkit +import org.bukkit.Material +import org.bukkit.NamespacedKey +import org.bukkit.craftbukkit.CraftServer +import org.bukkit.craftbukkit.entity.CraftEntity +import org.bukkit.craftbukkit.entity.CraftMob +import org.bukkit.craftbukkit.entity.CraftPlayer +import org.bukkit.craftbukkit.inventory.CraftItemStack +import org.bukkit.craftbukkit.inventory.CraftMetaArmor +import org.bukkit.craftbukkit.persistence.CraftPersistentDataContainer +import org.bukkit.craftbukkit.persistence.CraftPersistentDataTypeRegistry +import org.bukkit.craftbukkit.util.CraftMagicNumbers +import org.bukkit.craftbukkit.util.CraftNamespacedKey +import org.bukkit.entity.LivingEntity +import org.bukkit.entity.Mob +import org.bukkit.entity.Player +import org.bukkit.inventory.ItemStack +import org.bukkit.persistence.PersistentDataContainer +import java.lang.reflect.Field + +class CommonsInitializer : CommonsInitializerProxy { + override fun init(plugin: EcoPlugin) { + CommonsProvider.setIfNeeded(CommonsProviderImpl) + plugin.onEnable { + plugin.eventManager.registerListener(PacketInjectorListener) + } + } + + object CommonsProviderImpl : CommonsProvider { + private val cisHandle: Field = CraftItemStack::class.java.getDeclaredField("handle").apply { + isAccessible = true + } + + private val pdcRegsitry = CraftMetaArmor::class.java + .superclass // Access CraftMetaItem + .getDeclaredField("DATA_TYPE_REGISTRY") + .apply { isAccessible = true } + .get(null) as CraftPersistentDataTypeRegistry + + override val nbtTagString = CraftMagicNumbers.NBT.TAG_STRING + + override fun toPathfinderMob(mob: Mob): PathfinderMob? { + val craft = mob as? CraftMob ?: return null + return craft.handle as? PathfinderMob + } + + override fun toResourceLocation(namespacedKey: NamespacedKey): ResourceLocation = + CraftNamespacedKey.toMinecraft(namespacedKey) + + override fun asNMSStack(itemStack: ItemStack): net.minecraft.world.item.ItemStack { + return if (itemStack !is CraftItemStack) { + CraftItemStack.asNMSCopy(itemStack) + } else { + cisHandle[itemStack] as net.minecraft.world.item.ItemStack? ?: CraftItemStack.asNMSCopy(itemStack) + } + } + + override fun asBukkitStack(itemStack: net.minecraft.world.item.ItemStack): ItemStack { + return CraftItemStack.asCraftMirror(itemStack) + } + + override fun mergeIfNeeded(itemStack: ItemStack, nmsStack: net.minecraft.world.item.ItemStack) { + if (itemStack !is CraftItemStack) { + itemStack.itemMeta = CraftItemStack.asCraftMirror(nmsStack).itemMeta + } + } + + override fun toBukkitEntity(entity: net.minecraft.world.entity.LivingEntity): LivingEntity? = + CraftEntity.getEntity(Bukkit.getServer() as CraftServer, entity) as? LivingEntity + + override fun makePdc(tag: CompoundTag, base: Boolean): PersistentDataContainer { + fun emptyPdc(): CraftPersistentDataContainer = CraftPersistentDataContainer(pdcRegsitry) + + fun CompoundTag?.toPdc(): PersistentDataContainer { + val pdc = emptyPdc() + this ?: return pdc + val keys = this.allKeys + for (key in keys) { + pdc.put(key, this[key]) + } + + return pdc + } + + return if (base) { + tag.toPdc() + } else { + if (tag.contains("PublicBukkitValues")) { + tag.getCompound("PublicBukkitValues").toPdc() + } else { + emptyPdc() + } + } + } + + override fun setPdc( + tag: CompoundTag, + pdc: PersistentDataContainer?, + item: net.minecraft.world.item.ItemStack? + ) { + fun CraftPersistentDataContainer.toTag(): CompoundTag { + val compound = CompoundTag() + val rawPublicMap: Map = this.raw + for ((key, value) in rawPublicMap) { + compound.put(key, value) + } + + return compound + } + + val container = when (pdc) { + is CraftPersistentDataContainer? -> pdc + else -> null + } + + if (item != null) { + if (container != null && !container.isEmpty) { + for (key in tag.allKeys.toSet()) { + tag.remove(key) + } + + tag.merge(container.toTag()) + } else { + item.remove(DataComponents.CUSTOM_DATA) + } + } else { + if (container != null && !container.isEmpty) { + tag.put("PublicBukkitValues", container.toTag()) + } else { + tag.remove("PublicBukkitValues") + } + } + } + + override fun materialToItem(material: Material): Item = + BuiltInRegistries.ITEM.getOptional(material.key.toResourceLocation()) + .orElseThrow { IllegalArgumentException("Material is not item!") } + + override fun itemToMaterial(item: Item) = + Material.getMaterial(BuiltInRegistries.ITEM.getKey(item).path.uppercase()) + ?: throw IllegalArgumentException("Invalid material!") + + override fun toNMS(player: Player): ServerPlayer { + return (player as CraftPlayer).handle + } + + override fun toNMS(component: Component): net.minecraft.network.chat.Component { + val json = JSONComponentSerializer.json().serialize(component) + val holderLookupProvider = (Bukkit.getServer() as CraftServer).server.registryAccess() + + return net.minecraft.network.chat.Component.Serializer.fromJson(json, holderLookupProvider)!! + } + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/DisplayName.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/DisplayName.kt new file mode 100644 index 000000000..894cdbc97 --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/DisplayName.kt @@ -0,0 +1,57 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3 + +import com.willfp.eco.core.packet.Packet +import com.willfp.eco.core.packet.sendPacket +import com.willfp.eco.internal.spigot.proxy.DisplayNameProxy +import com.willfp.eco.internal.spigot.proxy.common.toNMS +import net.kyori.adventure.text.Component +import net.minecraft.network.protocol.game.ClientboundSetEntityDataPacket +import net.minecraft.network.syncher.EntityDataAccessor +import net.minecraft.network.syncher.SynchedEntityData +import net.minecraft.world.entity.Entity +import org.bukkit.craftbukkit.entity.CraftLivingEntity +import org.bukkit.entity.LivingEntity +import org.bukkit.entity.Player +import java.util.Optional + +@Suppress("UNCHECKED_CAST") +class DisplayName : DisplayNameProxy { + private val displayNameAccessor = Entity::class.java + .declaredFields + .filter { it.type == EntityDataAccessor::class.java } + .toList()[2] + .apply { isAccessible = true } + .get(null) as EntityDataAccessor> + + private val customNameVisibleAccessor = Entity::class.java + .declaredFields + .filter { it.type == EntityDataAccessor::class.java } + .toList()[3] + .apply { isAccessible = true } + .get(null) as EntityDataAccessor + + override fun setClientsideDisplayName( + entity: LivingEntity, + player: Player, + displayName: Component, + visible: Boolean + ) { + if (entity !is CraftLivingEntity) { + return + } + + val nmsComponent = displayName.toNMS() + + val nmsEntity = entity.handle + + val packet = ClientboundSetEntityDataPacket( + nmsEntity.id, + listOf( + SynchedEntityData.DataValue.create(displayNameAccessor, Optional.of(nmsComponent)), + SynchedEntityData.DataValue.create(customNameVisibleAccessor, visible) + ) + ) + + player.sendPacket(Packet(packet)) + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/DummyEntityFactory.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/DummyEntityFactory.kt new file mode 100644 index 000000000..5956457c8 --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/DummyEntityFactory.kt @@ -0,0 +1,15 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3 + +import com.willfp.eco.internal.entities.EcoDummyEntity +import com.willfp.eco.internal.spigot.proxy.DummyEntityFactoryProxy +import org.bukkit.Location +import org.bukkit.craftbukkit.CraftWorld +import org.bukkit.entity.Entity +import org.bukkit.entity.Zombie + +class DummyEntityFactory : DummyEntityFactoryProxy { + override fun createDummyEntity(location: Location): Entity { + val world = location.world as CraftWorld + return EcoDummyEntity(world.createEntity(location, Zombie::class.java)) + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/EntityControllerFactory.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/EntityControllerFactory.kt new file mode 100644 index 000000000..490316244 --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/EntityControllerFactory.kt @@ -0,0 +1,12 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3 + +import com.willfp.eco.core.entities.ai.EntityController +import com.willfp.eco.internal.spigot.proxy.EntityControllerFactoryProxy +import com.willfp.eco.internal.spigot.proxy.v1_21_3.entity.EcoEntityController +import org.bukkit.entity.Mob + +class EntityControllerFactory : EntityControllerFactoryProxy { + override fun createEntityController(entity: T): EntityController { + return EcoEntityController(entity) + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/ExtendedPersistentDataContainerFactory.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/ExtendedPersistentDataContainerFactory.kt new file mode 100644 index 000000000..16a6a7421 --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/ExtendedPersistentDataContainerFactory.kt @@ -0,0 +1,91 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3 + +import com.willfp.eco.core.data.ExtendedPersistentDataContainer +import com.willfp.eco.internal.spigot.proxy.ExtendedPersistentDataContainerFactoryProxy +import net.minecraft.nbt.Tag +import org.bukkit.Material +import org.bukkit.craftbukkit.inventory.CraftItemStack +import org.bukkit.craftbukkit.persistence.CraftPersistentDataContainer +import org.bukkit.craftbukkit.persistence.CraftPersistentDataTypeRegistry +import org.bukkit.inventory.ItemStack +import org.bukkit.persistence.PersistentDataContainer +import org.bukkit.persistence.PersistentDataType +import java.lang.reflect.Field + +class ExtendedPersistentDataContainerFactory : ExtendedPersistentDataContainerFactoryProxy { + private val registry: CraftPersistentDataTypeRegistry + + init { + /* + Can't grab actual instance since it's in CraftMetaItem (which is package-private) + And getting it would mean more janky reflection + */ + val item = CraftItemStack.asCraftCopy(ItemStack(Material.STONE)) + val pdc = item.itemMeta!!.persistentDataContainer + + // Cross-version compatibility: + val registryField: Field = try { + CraftPersistentDataContainer::class.java.getDeclaredField("registry") + } catch (e: NoSuchFieldException) { + CraftPersistentDataContainer::class.java.superclass.getDeclaredField("registry") + } + + this.registry = registryField + .apply { isAccessible = true }.get(pdc) as CraftPersistentDataTypeRegistry + } + + override fun adapt(pdc: PersistentDataContainer): ExtendedPersistentDataContainer { + return when (pdc) { + is CraftPersistentDataContainer -> EcoPersistentDataContainer(pdc) + else -> throw IllegalArgumentException("Custom PDC instance ${pdc::class.java.name} is not supported!") + } + } + + override fun newPdc(): PersistentDataContainer { + return CraftPersistentDataContainer(registry) + } + + inner class EcoPersistentDataContainer( + private val handle: CraftPersistentDataContainer + ) : ExtendedPersistentDataContainer { + @Suppress("UNCHECKED_CAST") + private val customDataTags: MutableMap = + CraftPersistentDataContainer::class.java.getDeclaredField("customDataTags") + .apply { isAccessible = true }.get(handle) as MutableMap + + override fun set(key: String, dataType: PersistentDataType, value: Z) { + customDataTags[key] = + registry.wrap(dataType, dataType.toPrimitive(value, handle.adapterContext)) + } + + override fun has(key: String, dataType: PersistentDataType): Boolean { + val value = customDataTags[key] ?: return false + return registry.isInstanceOf(dataType, value) + } + + override fun get(key: String, dataType: PersistentDataType): Z? { + val value = customDataTags[key] ?: return null + return dataType.fromPrimitive(registry.extract(dataType, value), handle.adapterContext) + } + + override fun getOrDefault( + key: String, + dataType: PersistentDataType, + defaultValue: Z + ): Z { + return get(key, dataType) ?: defaultValue + } + + override fun remove(key: String) { + customDataTags.remove(key) + } + + override fun getAllKeys(): MutableSet { + return customDataTags.keys + } + + override fun getBase(): PersistentDataContainer { + return handle + } + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/FastItemStackFactory.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/FastItemStackFactory.kt new file mode 100644 index 000000000..dce32a476 --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/FastItemStackFactory.kt @@ -0,0 +1,12 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3 + +import com.willfp.eco.core.fast.FastItemStack +import com.willfp.eco.internal.spigot.proxy.FastItemStackFactoryProxy +import com.willfp.eco.internal.spigot.proxy.common.modern.NewEcoFastItemStack +import org.bukkit.inventory.ItemStack + +class FastItemStackFactory : FastItemStackFactoryProxy { + override fun create(itemStack: ItemStack): FastItemStack { + return NewEcoFastItemStack(itemStack) + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/MiniMessageTranslator.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/MiniMessageTranslator.kt new file mode 100644 index 000000000..3b45b27e8 --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/MiniMessageTranslator.kt @@ -0,0 +1,33 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3 + +import com.willfp.eco.core.display.Display +import com.willfp.eco.internal.spigot.proxy.MiniMessageTranslatorProxy +import com.willfp.eco.util.toLegacy +import net.kyori.adventure.text.minimessage.MiniMessage + +class MiniMessageTranslator : MiniMessageTranslatorProxy { + override fun format(message: String): String { + var mut = message + + val startsWithPrefix = mut.startsWith(Display.PREFIX) + if (startsWithPrefix) { + mut = mut.substring(2) + } + + mut = mut.replace('ยง', '&') + + val miniMessage = runCatching { + MiniMessage.miniMessage().deserialize( + mut + ).toLegacy() + }.getOrNull() ?: mut + + mut = if (startsWithPrefix) { + Display.PREFIX + miniMessage + } else { + miniMessage + } + + return mut + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/PacketHandler.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/PacketHandler.kt new file mode 100644 index 000000000..093052cf7 --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/PacketHandler.kt @@ -0,0 +1,47 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3 + +import com.willfp.eco.core.EcoPlugin +import com.willfp.eco.core.packet.PacketListener +import com.willfp.eco.internal.spigot.proxy.PacketHandlerProxy +import com.willfp.eco.internal.spigot.proxy.common.packet.display.PacketAutoRecipe +import com.willfp.eco.internal.spigot.proxy.common.packet.display.PacketHeldItemSlot +import com.willfp.eco.internal.spigot.proxy.common.packet.display.PacketSetSlot +import com.willfp.eco.internal.spigot.proxy.common.packet.display.PacketWindowItems +import com.willfp.eco.internal.spigot.proxy.common.packet.display.frame.clearFrames +import com.willfp.eco.internal.spigot.proxy.v1_21_3.packet.NewItemsPacketOpenWindowMerchant +import com.willfp.eco.internal.spigot.proxy.v1_21_3.packet.NewItemsPacketSetCreativeSlot +import net.minecraft.network.protocol.Packet +import org.bukkit.craftbukkit.entity.CraftPlayer +import org.bukkit.entity.Player + +class PacketHandler : PacketHandlerProxy { + override fun sendPacket(player: Player, packet: com.willfp.eco.core.packet.Packet) { + if (player !is CraftPlayer) { + return + } + + val handle = packet.handle + + if (handle !is Packet<*>) { + return + } + + player.handle.connection.send(handle) + } + + override fun clearDisplayFrames() { + clearFrames() + } + + override fun getPacketListeners(plugin: EcoPlugin): List { + // No PacketAutoRecipe for 1.21.3+ because recipes have been changed internally + + return listOf( + PacketHeldItemSlot, + NewItemsPacketOpenWindowMerchant, + NewItemsPacketSetCreativeSlot, + PacketSetSlot, + PacketWindowItems(plugin) + ) + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/SNBTConverter.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/SNBTConverter.kt new file mode 100644 index 000000000..58b6bf05a --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/SNBTConverter.kt @@ -0,0 +1,80 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3 + +import com.mojang.serialization.Dynamic +import com.willfp.eco.core.items.TestableItem +import com.willfp.eco.core.recipe.parts.EmptyTestableItem +import com.willfp.eco.internal.spigot.proxy.SNBTConverterProxy +import net.minecraft.nbt.CompoundTag +import net.minecraft.nbt.NbtOps +import net.minecraft.nbt.SnbtPrinterTagVisitor +import net.minecraft.nbt.TagParser +import net.minecraft.server.MinecraftServer +import net.minecraft.util.datafix.fixes.References +import org.bukkit.Bukkit +import org.bukkit.craftbukkit.CraftServer +import org.bukkit.craftbukkit.inventory.CraftItemStack +import org.bukkit.craftbukkit.util.CraftMagicNumbers +import org.bukkit.inventory.ItemStack + +private val registryAccess = (Bukkit.getServer() as CraftServer).server.registryAccess() + +class SNBTConverter : SNBTConverterProxy { + private fun parseItemSNBT(snbt: String): CompoundTag? { + val nbt = runCatching { TagParser.parseTag(snbt) }.getOrNull() ?: return null + val dataVersion = if (nbt.contains("DataVersion")) { + nbt.getInt("DataVersion") + } else null + + // If the data version is the same as the server's data version, we don't need to fix it + if (dataVersion == CraftMagicNumbers.INSTANCE.dataVersion) { + return nbt + } + + return MinecraftServer.getServer().fixerUpper.update( + References.ITEM_STACK, + Dynamic(NbtOps.INSTANCE, nbt), + dataVersion ?: 3700, // 3700 is the 1.20.4 data version + CraftMagicNumbers.INSTANCE.dataVersion + ).value as CompoundTag + } + + override fun fromSNBT(snbt: String): ItemStack? { + val tag = parseItemSNBT(snbt) ?: return null + val nms = net.minecraft.world.item.ItemStack.parse(registryAccess, tag).orElse(null) ?: return null + return CraftItemStack.asBukkitCopy(nms) + } + + override fun toSNBT(itemStack: ItemStack): String { + val nms = CraftItemStack.asNMSCopy(itemStack) + val tag = nms.save(registryAccess) as CompoundTag + tag.putInt("DataVersion", CraftMagicNumbers.INSTANCE.dataVersion) + return SnbtPrinterTagVisitor().visit(tag) + } + + override fun makeSNBTTestable(snbt: String): TestableItem { + val tag = parseItemSNBT(snbt) ?: return EmptyTestableItem() + val nms = net.minecraft.world.item.ItemStack.parse(registryAccess, tag).orElse(null) + ?: return EmptyTestableItem() + + tag.remove("Count") + return SNBTTestableItem(CraftItemStack.asBukkitCopy(nms), tag) + } + + class SNBTTestableItem( + private val item: ItemStack, + private val tag: CompoundTag + ) : TestableItem { + override fun matches(itemStack: ItemStack?): Boolean { + if (itemStack == null) { + return false + } + + val nms = CraftItemStack.asNMSCopy(itemStack) + val nmsTag = nms.save(registryAccess) as CompoundTag + nmsTag.remove("Count") + return tag.copy().merge(nmsTag) == nmsTag && itemStack.type == item.type + } + + override fun getItem(): ItemStack = item + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/Skull.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/Skull.kt new file mode 100644 index 000000000..7129afd19 --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/Skull.kt @@ -0,0 +1,18 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3 + +import com.willfp.eco.internal.spigot.proxy.SkullProxy +import com.willfp.eco.internal.spigot.proxy.common.modern.texture +import org.bukkit.inventory.meta.SkullMeta + +class Skull : SkullProxy { + override fun setSkullTexture( + meta: SkullMeta, + base64: String + ) { + meta.texture = base64 + } + + override fun getSkullTexture( + meta: SkullMeta + ): String? = meta.texture +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/TPS.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/TPS.kt new file mode 100644 index 000000000..ed4dde5ce --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/TPS.kt @@ -0,0 +1,11 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3 + +import com.willfp.eco.internal.spigot.proxy.TPSProxy +import org.bukkit.Bukkit +import org.bukkit.craftbukkit.CraftServer + +class TPS : TPSProxy { + override fun getTPS(): Double { + return (Bukkit.getServer() as CraftServer).handle.server.tps1.average + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/entity/EcoEntityController.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/entity/EcoEntityController.kt new file mode 100644 index 000000000..2b0eb93ad --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/entity/EcoEntityController.kt @@ -0,0 +1,95 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3.entity + +import com.willfp.eco.core.entities.ai.CustomGoal +import com.willfp.eco.core.entities.ai.EntityController +import com.willfp.eco.core.entities.ai.EntityGoal +import com.willfp.eco.core.entities.ai.TargetGoal +import com.willfp.eco.internal.spigot.proxy.common.ai.CustomGoalFactory +import com.willfp.eco.internal.spigot.proxy.common.ai.getGoalFactory +import com.willfp.eco.internal.spigot.proxy.common.toPathfinderMob +import net.minecraft.world.entity.PathfinderMob +import net.minecraft.world.entity.ai.goal.Goal +import org.bukkit.entity.Mob + +class EcoEntityController( + private val handle: T +) : EntityController { + override fun addEntityGoal(priority: Int, goal: EntityGoal): EntityController { + val nms = getNms() ?: return this + + nms.goalSelector.addGoal( + priority, + goal.getGoalFactory()?.create(goal, nms) ?: return this + ) + + return this + } + + override fun removeEntityGoal(goal: EntityGoal): EntityController { + val nms = getNms() ?: return this + + val predicate: (Goal) -> Boolean = if (goal is CustomGoal<*>) { + { CustomGoalFactory.isGoalOfType(it, goal) } + } else { + { goal.getGoalFactory()?.isGoalOfType(it) == true } + } + + for (wrapped in nms.goalSelector.availableGoals.toSet()) { + if (predicate(wrapped.goal)) { + nms.goalSelector.removeGoal(wrapped.goal) + } + } + + return this + } + + override fun clearEntityGoals(): EntityController { + val nms = getNms() ?: return this + nms.goalSelector.availableGoals.clear() + return this + } + + override fun addTargetGoal(priority: Int, goal: TargetGoal): EntityController { + val nms = getNms() ?: return this + + nms.targetSelector.addGoal( + priority, goal.getGoalFactory()?.create(goal, nms) ?: return this + ) + + nms.targetSelector + + return this + } + + override fun removeTargetGoal(goal: TargetGoal): EntityController { + val nms = getNms() ?: return this + + val predicate: (Goal) -> Boolean = if (goal is CustomGoal<*>) { + { CustomGoalFactory.isGoalOfType(it, goal) } + } else { + { goal.getGoalFactory()?.isGoalOfType(it) == true } + } + + for (wrapped in nms.targetSelector.availableGoals.toSet()) { + if (predicate(wrapped.goal)) { + nms.targetSelector.removeGoal(wrapped.goal) + } + } + + return this + } + + override fun clearTargetGoals(): EntityController { + val nms = getNms() ?: return this + nms.targetSelector.availableGoals.clear() + return this + } + + private fun getNms(): PathfinderMob? { + return handle.toPathfinderMob() + } + + override fun getEntity(): T { + return handle + } +} \ No newline at end of file diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/packet/NewItemsPacketOpenWindowMerchant.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/packet/NewItemsPacketOpenWindowMerchant.kt new file mode 100644 index 000000000..3bcccfbe6 --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/packet/NewItemsPacketOpenWindowMerchant.kt @@ -0,0 +1,35 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3.packet + +import com.willfp.eco.core.display.Display +import com.willfp.eco.core.packet.PacketEvent +import com.willfp.eco.core.packet.PacketListener +import com.willfp.eco.internal.spigot.proxy.common.asBukkitStack +import net.minecraft.network.protocol.game.ClientboundMerchantOffersPacket +import net.minecraft.world.item.trading.MerchantOffers + +object NewItemsPacketOpenWindowMerchant : PacketListener { + private val field = ClientboundMerchantOffersPacket::class.java + .declaredFields + .first { it.type == MerchantOffers::class.java } + .apply { isAccessible = true } + + override fun onSend(event: PacketEvent) { + val packet = event.packet.handle as? ClientboundMerchantOffersPacket ?: return + + val offers = MerchantOffers() + + for (offer in packet.offers) { + val new = offer.copy() + + Display.display(new.baseCostA.itemStack.asBukkitStack(), event.player) + if (new.costB.isPresent) { + Display.display(new.costB.get().itemStack.asBukkitStack(), event.player) + } + Display.display(new.result.asBukkitStack(), event.player) + + offers += new + } + + field.set(packet, offers) + } +} diff --git a/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/packet/NewItemsPacketSetCreativeSlot.kt b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/packet/NewItemsPacketSetCreativeSlot.kt new file mode 100644 index 000000000..55a467b9d --- /dev/null +++ b/eco-core/core-nms/v1_21_3/src/main/kotlin/com/willfp/eco/internal/spigot/proxy/v1_21_3/packet/NewItemsPacketSetCreativeSlot.kt @@ -0,0 +1,19 @@ +package com.willfp.eco.internal.spigot.proxy.v1_21_3.packet + +import com.willfp.eco.core.display.Display +import com.willfp.eco.core.packet.PacketEvent +import com.willfp.eco.core.packet.PacketListener +import com.willfp.eco.internal.spigot.proxy.common.asBukkitStack +import com.willfp.eco.internal.spigot.proxy.common.packet.display.frame.DisplayFrame +import com.willfp.eco.internal.spigot.proxy.common.packet.display.frame.lastDisplayFrame +import net.minecraft.network.protocol.game.ServerboundSetCreativeModeSlotPacket + +object NewItemsPacketSetCreativeSlot : PacketListener { + override fun onReceive(event: PacketEvent) { + val packet = event.packet.handle as? ServerboundSetCreativeModeSlotPacket ?: return + + Display.revert(packet.itemStack.asBukkitStack()) + + event.player.lastDisplayFrame = DisplayFrame.EMPTY + } +} diff --git a/eco-core/core-plugin/src/main/kotlin/com/willfp/eco/internal/spigot/recipes/CraftingRecipeListener.kt b/eco-core/core-plugin/src/main/kotlin/com/willfp/eco/internal/spigot/recipes/CraftingRecipeListener.kt index 53dc08a68..b927ad089 100644 --- a/eco-core/core-plugin/src/main/kotlin/com/willfp/eco/internal/spigot/recipes/CraftingRecipeListener.kt +++ b/eco-core/core-plugin/src/main/kotlin/com/willfp/eco/internal/spigot/recipes/CraftingRecipeListener.kt @@ -1,7 +1,9 @@ package com.willfp.eco.internal.spigot.recipes import com.willfp.eco.core.EcoPlugin +import com.willfp.eco.core.Prerequisite import com.willfp.eco.core.recipe.Recipes +import com.willfp.eco.util.namespacedKeyOf import org.bukkit.Keyed import org.bukkit.event.EventHandler import org.bukkit.event.Listener @@ -11,7 +13,11 @@ import org.bukkit.event.player.PlayerRecipeDiscoverEvent class CraftingRecipeListener(val plugin: EcoPlugin) : Listener { @EventHandler - fun preventLearningDisplayedRecipes(event: PlayerRecipeDiscoverEvent) { + fun handleDisplayedRecipeUnlocksPre1213(event: PlayerRecipeDiscoverEvent) { + if (Prerequisite.HAS_1_21_3.isMet) { + return + } + if (!EcoPlugin.getPluginNames().contains(event.recipe.namespace)) { return } @@ -21,6 +27,27 @@ class CraftingRecipeListener(val plugin: EcoPlugin) : Listener { } } + @EventHandler + fun handleDisplayedRecipeUnlocks1213(event: PlayerRecipeDiscoverEvent) { + if (!Prerequisite.HAS_1_21_3.isMet) { + return + } + + if (!EcoPlugin.getPluginNames().contains(event.recipe.namespace)) { + return + } + + if (!event.recipe.key.contains("_displayed")) { + event.isCancelled = true + + val player = event.player + player.discoverRecipe(namespacedKeyOf( + event.recipe.namespace, + event.recipe.key + "_displayed" + )) + } + } + @EventHandler fun processListeners(event: PrepareItemCraftEvent) { handlePrepare(event) diff --git a/gradle.properties b/gradle.properties index 8cb3adcdd..16b80b5b5 100644 --- a/gradle.properties +++ b/gradle.properties @@ -1,2 +1,2 @@ -version = 6.74.2 +version = 6.74.3 kotlin.incremental.useClasspathSnapshot=false \ No newline at end of file diff --git a/settings.gradle.kts b/settings.gradle.kts index 48786c739..e686002a8 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -27,6 +27,7 @@ include(":eco-core:core-nms:v1_20_R1") include(":eco-core:core-nms:v1_20_R2") include(":eco-core:core-nms:v1_20_R3") include(":eco-core:core-nms:v1_21") +include(":eco-core:core-nms:v1_21_3") include(":eco-core:core-proxy") include(":eco-core:core-plugin") include(":eco-core:core-backend")