mirror of
https://github.com/Xiao-MoMi/craft-engine.git
synced 2025-12-19 15:09:15 +00:00
太难写了
This commit is contained in:
@@ -5,14 +5,12 @@ import net.kyori.adventure.text.Component;
|
||||
import net.momirealms.craftengine.bukkit.api.BukkitAdaptors;
|
||||
import net.momirealms.craftengine.bukkit.item.BukkitItemManager;
|
||||
import net.momirealms.craftengine.bukkit.item.ComponentTypes;
|
||||
import net.momirealms.craftengine.bukkit.nms.FastNMS;
|
||||
import net.momirealms.craftengine.bukkit.plugin.BukkitCraftEngine;
|
||||
import net.momirealms.craftengine.bukkit.plugin.reflection.bukkit.CraftBukkitReflections;
|
||||
import net.momirealms.craftengine.bukkit.plugin.reflection.minecraft.CoreReflections;
|
||||
import net.momirealms.craftengine.bukkit.plugin.user.BukkitServerPlayer;
|
||||
import net.momirealms.craftengine.bukkit.util.ComponentUtils;
|
||||
import net.momirealms.craftengine.bukkit.util.InventoryUtils;
|
||||
import net.momirealms.craftengine.bukkit.util.ItemStackUtils;
|
||||
import net.momirealms.craftengine.bukkit.util.LegacyInventoryUtils;
|
||||
import net.momirealms.craftengine.bukkit.util.*;
|
||||
import net.momirealms.craftengine.core.item.*;
|
||||
import net.momirealms.craftengine.core.item.equipment.TrimBasedEquipment;
|
||||
import net.momirealms.craftengine.core.item.recipe.*;
|
||||
@@ -36,6 +34,7 @@ import org.bukkit.event.inventory.*;
|
||||
import org.bukkit.event.inventory.ClickType;
|
||||
import org.bukkit.inventory.*;
|
||||
import org.bukkit.inventory.view.AnvilView;
|
||||
import org.jetbrains.annotations.Nullable;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
@@ -599,15 +598,14 @@ public class RecipeEventListener implements Listener {
|
||||
@EventHandler(ignoreCancelled = true)
|
||||
public void onPrepareCraftingRecipe(PrepareItemCraftEvent event) {
|
||||
if (!Config.enableRecipeSystem()) return;
|
||||
org.bukkit.inventory.Recipe recipe = event.getRecipe();
|
||||
if (!(recipe instanceof CraftingRecipe craftingRecipe)) return;
|
||||
Key recipeId = Key.of(craftingRecipe.getKey().namespace(), craftingRecipe.getKey().value());
|
||||
CraftingInventory inventory = event.getInventory();
|
||||
Key recipeId = getCurrentCraftingRecipeId(inventory);
|
||||
if (recipeId == null) return;
|
||||
Optional<Recipe<ItemStack>> optionalRecipe = this.recipeManager.recipeById(recipeId);
|
||||
// 也许是其他插件注册的配方,直接无视
|
||||
if (optionalRecipe.isEmpty()) {
|
||||
return;
|
||||
}
|
||||
CraftingInventory inventory = event.getInventory();
|
||||
if (!(optionalRecipe.get() instanceof CustomCraftingTableRecipe<ItemStack> craftingTableRecipe)) {
|
||||
inventory.setResult(null);
|
||||
return;
|
||||
@@ -634,39 +632,123 @@ public class RecipeEventListener implements Listener {
|
||||
@EventHandler(ignoreCancelled = true, priority = EventPriority.HIGHEST)
|
||||
public void onCraftingFinish(CraftItemEvent event) {
|
||||
if (!Config.enableRecipeSystem() || !VersionHelper.PREMIUM) return;
|
||||
Event.Result result = event.getResult();
|
||||
org.bukkit.inventory.Recipe recipe = event.getRecipe();
|
||||
if (!(recipe instanceof CraftingRecipe craftingRecipe)) return;
|
||||
Key recipeId = Key.of(craftingRecipe.getKey().namespace(), craftingRecipe.getKey().value());
|
||||
CraftingInventory inventory = event.getInventory();
|
||||
ItemStack visualResultOrReal = inventory.getResult();
|
||||
// 可惜我们没有结果
|
||||
if (ItemStackUtils.isEmpty(visualResultOrReal)) return;
|
||||
Key recipeId = getCurrentCraftingRecipeId(inventory);
|
||||
if (recipeId == null) return;
|
||||
Optional<Recipe<ItemStack>> optionalRecipe = this.recipeManager.recipeById(recipeId);
|
||||
// 也许是其他插件注册的配方,直接无视
|
||||
if (optionalRecipe.isEmpty()) {
|
||||
if (optionalRecipe.isEmpty() || !(optionalRecipe.get() instanceof CustomCraftingTableRecipe<ItemStack> ceRecipe)) {
|
||||
return;
|
||||
}
|
||||
CraftingInventory inventory = event.getInventory();
|
||||
if (ItemStackUtils.isEmpty(inventory.getResult())) return;
|
||||
if (!(optionalRecipe.get() instanceof CustomCraftingTableRecipe<ItemStack> craftingTableRecipe)) {
|
||||
// 没有视觉结果和函数你凑什么热闹
|
||||
if (!ceRecipe.hasVisualResult() && !ceRecipe.hasFunctions()) {
|
||||
return;
|
||||
}
|
||||
// 无事发生,不要更新
|
||||
if (event.getAction() == InventoryAction.NOTHING) {
|
||||
return;
|
||||
}
|
||||
Player player = InventoryUtils.getPlayerFromInventoryEvent(event);
|
||||
BukkitServerPlayer serverPlayer = BukkitAdaptors.adapt(player);
|
||||
// todo shift点击应该更好地处理,包括function执行次数也是
|
||||
if (craftingTableRecipe.hasVisualResult()) {
|
||||
// 多次合成
|
||||
if (event.isShiftClick()) {
|
||||
// 由插件自己处理多次合成
|
||||
event.setResult(Event.Result.DENY);
|
||||
|
||||
Object mcPlayer = serverPlayer.serverPlayer();
|
||||
Object craftingMenu = FastNMS.INSTANCE.field$Player$containerMenu(mcPlayer);
|
||||
|
||||
// 如果有视觉结果,先临时替换为真实的
|
||||
if (ceRecipe.hasVisualResult()) {
|
||||
inventory.setResult(ceRecipe.assemble(null, ItemBuildContext.of(serverPlayer)));
|
||||
}
|
||||
// 先取一次
|
||||
Object itemMoved = FastNMS.INSTANCE.method$AbstractContainerMenu$quickMoveStack(craftingMenu, mcPlayer, 0 /* result slot */);
|
||||
if (FastNMS.INSTANCE.method$ItemStack$isEmpty(itemMoved)) {
|
||||
// 发现取了个寂寞,根本没地方放,给他复原成视觉结果
|
||||
inventory.setResult(visualResultOrReal);
|
||||
return;
|
||||
}
|
||||
// 有函数的情况下,执行函数
|
||||
if (ceRecipe.hasFunctions()) {
|
||||
PlayerOptionalContext context = PlayerOptionalContext.of(serverPlayer);
|
||||
for (Function<PlayerOptionalContext> function : ceRecipe.functions()) {
|
||||
function.run(context);
|
||||
}
|
||||
}
|
||||
|
||||
for (;;) {
|
||||
// 这个时候配方已经更新了,如果变化了,那么就不要操作
|
||||
if (!recipeId.equals(getCurrentCraftingRecipeId(inventory))) {
|
||||
break;
|
||||
}
|
||||
|
||||
// 配方不变,允许起飞
|
||||
// 如果有视觉结果,先临时替换为真实的
|
||||
if (ceRecipe.hasVisualResult()) {
|
||||
inventory.setResult(ceRecipe.assemble(null, ItemBuildContext.of(serverPlayer)));
|
||||
}
|
||||
|
||||
// 连续获取
|
||||
itemMoved = FastNMS.INSTANCE.method$AbstractContainerMenu$quickMoveStack(craftingMenu, mcPlayer, 0 /* result slot */);
|
||||
if (FastNMS.INSTANCE.method$ItemStack$isEmpty(itemMoved)) {
|
||||
// 发现取了个寂寞,根本没地方放,给他复原成视觉结果
|
||||
inventory.setResult(visualResultOrReal);
|
||||
break;
|
||||
}
|
||||
// 有函数的情况下,执行函数
|
||||
if (ceRecipe.hasFunctions()) {
|
||||
PlayerOptionalContext context = PlayerOptionalContext.of(serverPlayer);
|
||||
for (Function<PlayerOptionalContext> function : ceRecipe.functions()) {
|
||||
function.run(context);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
// 单次合成
|
||||
else {
|
||||
// 指针物品不为空,且竟然和视觉物品一致,逆天,必须阻止
|
||||
if (event.getClick() == ClickType.LEFT || event.getClick() == ClickType.RIGHT) {
|
||||
ItemStack cursor = event.getCursor();
|
||||
if (!ItemStackUtils.isEmpty(cursor)) {
|
||||
if (cursor.isSimilar(visualResultOrReal)) {
|
||||
event.setResult(Event.Result.DENY);
|
||||
return;
|
||||
}
|
||||
CraftingInput<ItemStack> input = getCraftingInput(inventory);
|
||||
inventory.setResult(craftingTableRecipe.assemble(input, ItemBuildContext.of(serverPlayer)));
|
||||
}
|
||||
Function<PlayerOptionalContext>[] functions = craftingTableRecipe.functions();
|
||||
if (functions != null) {
|
||||
}
|
||||
// 有视觉结果的情况下,重新构造真实物品
|
||||
if (ceRecipe.hasVisualResult()) {
|
||||
inventory.setResult(ceRecipe.assemble(null, ItemBuildContext.of(serverPlayer)));
|
||||
}
|
||||
// 有函数的情况下,执行函数
|
||||
if (ceRecipe.hasFunctions()) {
|
||||
PlayerOptionalContext context = PlayerOptionalContext.of(serverPlayer);
|
||||
for (Function<PlayerOptionalContext> function : functions) {
|
||||
for (Function<PlayerOptionalContext> function : ceRecipe.functions()) {
|
||||
function.run(context);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// bukkit的getRecipe会生成新的recipe对象,过程较慢,只需要获取配方id即可
|
||||
@Nullable
|
||||
private Key getCurrentCraftingRecipeId(CraftingInventory inventory) {
|
||||
Object craftContainer = FastNMS.INSTANCE.method$CraftInventory$getInventory(inventory);
|
||||
Object recipeHolderOrRecipe = FastNMS.INSTANCE.method$CraftingContainer$getCurrentRecipe(craftContainer);
|
||||
if (recipeHolderOrRecipe == null) return null;
|
||||
if (VersionHelper.isOrAbove1_21_2()) {
|
||||
return KeyUtils.resourceLocationToKey(FastNMS.INSTANCE.field$ResourceKey$location(FastNMS.INSTANCE.field$RecipeHolder$id(recipeHolderOrRecipe)));
|
||||
} else if (VersionHelper.isOrAbove1_20_2()) {
|
||||
return KeyUtils.resourceLocationToKey(FastNMS.INSTANCE.field$RecipeHolder$id(recipeHolderOrRecipe));
|
||||
} else {
|
||||
// 其实是recipe getId的实现
|
||||
return KeyUtils.resourceLocationToKey(FastNMS.INSTANCE.field$RecipeHolder$id(recipeHolderOrRecipe));
|
||||
}
|
||||
}
|
||||
|
||||
private CraftingInput<ItemStack> getCraftingInput(CraftingInventory inventory) {
|
||||
ItemStack[] ingredients = inventory.getMatrix();
|
||||
@@ -765,6 +847,8 @@ public class RecipeEventListener implements Listener {
|
||||
return;
|
||||
}
|
||||
SmithingInventory inventory = event.getInventory();
|
||||
ItemStack result = inventory.getResult();
|
||||
if (ItemStackUtils.isEmpty(result)) return;
|
||||
Player player = InventoryUtils.getPlayerFromInventoryEvent(event);
|
||||
BukkitServerPlayer serverPlayer = BukkitAdaptors.adapt(player);
|
||||
if (smithingRecipe.hasVisualResult()) {
|
||||
@@ -772,6 +856,19 @@ public class RecipeEventListener implements Listener {
|
||||
event.setCancelled(true);
|
||||
return;
|
||||
}
|
||||
if (event.getAction() == InventoryAction.NOTHING) {
|
||||
return;
|
||||
}
|
||||
// 指针物品不为空
|
||||
if (event.getClick() == ClickType.LEFT || event.getClick() == ClickType.RIGHT) {
|
||||
ItemStack cursor = event.getCursor();
|
||||
if (!ItemStackUtils.isEmpty(cursor)) {
|
||||
if (cursor.isSimilar(result)) {
|
||||
event.setResult(Event.Result.DENY);
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
SmithingInput<ItemStack> input = getSmithingInput(inventory);
|
||||
inventory.setResult(smithingRecipe.assemble(input, ItemBuildContext.of(serverPlayer)));
|
||||
}
|
||||
@@ -788,10 +885,30 @@ public class RecipeEventListener implements Listener {
|
||||
if (optionalRecipe.isEmpty() || !(optionalRecipe.get() instanceof CustomSmithingTrimRecipe<ItemStack> smithingRecipe)) {
|
||||
return;
|
||||
}
|
||||
SmithingInventory inventory = event.getInventory();
|
||||
ItemStack result = inventory.getResult();
|
||||
if (ItemStackUtils.isEmpty(result)) return;
|
||||
Player player = InventoryUtils.getPlayerFromInventoryEvent(event);
|
||||
BukkitServerPlayer serverPlayer = BukkitAdaptors.adapt(player);
|
||||
Function<PlayerOptionalContext>[] functions = smithingRecipe.functions();
|
||||
if (functions != null) {
|
||||
if (event.isShiftClick()) {
|
||||
event.setCancelled(true);
|
||||
return;
|
||||
}
|
||||
if (event.getAction() == InventoryAction.NOTHING) {
|
||||
return;
|
||||
}
|
||||
// 指针物品不为空
|
||||
if (event.getClick() == ClickType.LEFT || event.getClick() == ClickType.RIGHT) {
|
||||
ItemStack cursor = event.getCursor();
|
||||
if (!ItemStackUtils.isEmpty(cursor)) {
|
||||
if (cursor.isSimilar(result)) {
|
||||
event.setResult(Event.Result.DENY);
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
PlayerOptionalContext context = PlayerOptionalContext.of(serverPlayer);
|
||||
for (Function<PlayerOptionalContext> function : functions) {
|
||||
function.run(context);
|
||||
|
||||
@@ -3487,10 +3487,10 @@ public class BukkitNetworkManager implements NetworkManager, Listener, PluginMes
|
||||
buf.writeByte(buttonNum);
|
||||
buf.writeVarInt(clickType);
|
||||
buf.writeVarInt(changedSlots.size());
|
||||
changedSlots.forEach((k, v) -> {
|
||||
buf.writeShort(k);
|
||||
FastNMS.INSTANCE.method$FriendlyByteBuf$writeItem(friendlyBuf, v);
|
||||
});
|
||||
for (Map.Entry<Integer, ItemStack> entry : changedSlots.int2ObjectEntrySet()) {
|
||||
buf.writeShort(entry.getKey());
|
||||
FastNMS.INSTANCE.method$FriendlyByteBuf$writeItem(friendlyBuf, entry.getValue());
|
||||
}
|
||||
FastNMS.INSTANCE.method$FriendlyByteBuf$writeItem(friendlyBuf, carriedItem);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -5,6 +5,10 @@ import net.momirealms.craftengine.core.plugin.context.function.Function;
|
||||
|
||||
public interface FunctionalRecipe<T> extends Recipe<T> {
|
||||
|
||||
default boolean hasFunctions() {
|
||||
Function<PlayerOptionalContext>[] functions = functions();
|
||||
return functions != null && functions.length > 0;
|
||||
}
|
||||
|
||||
Function<PlayerOptionalContext>[] functions();
|
||||
}
|
||||
|
||||
@@ -50,7 +50,7 @@ byte_buddy_version=1.17.8
|
||||
ahocorasick_version=0.6.3
|
||||
snake_yaml_version=2.5
|
||||
anti_grief_version=1.0.3
|
||||
nms_helper_version=1.0.114
|
||||
nms_helper_version=1.0.117
|
||||
evalex_version=3.5.0
|
||||
reactive_streams_version=1.0.4
|
||||
amazon_awssdk_version=2.34.5
|
||||
|
||||
Reference in New Issue
Block a user