mirror of
https://github.com/ViaVersion/ViaBackwards.git
synced 2024-11-21 12:07:38 +01:00
Start working on recipes
This commit is contained in:
parent
d3be34adc9
commit
4937ec4800
@ -24,8 +24,8 @@ import com.viaversion.viaversion.protocols.v1_21to1_21_2.packet.ClientboundPacke
|
|||||||
import com.viaversion.viaversion.protocols.v1_21to1_21_2.packet.ClientboundPackets1_21_2;
|
import com.viaversion.viaversion.protocols.v1_21to1_21_2.packet.ClientboundPackets1_21_2;
|
||||||
import com.viaversion.viaversion.protocols.v1_21to1_21_2.packet.ServerboundPacket1_21_2;
|
import com.viaversion.viaversion.protocols.v1_21to1_21_2.packet.ServerboundPacket1_21_2;
|
||||||
import com.viaversion.viaversion.protocols.v1_21to1_21_2.packet.ServerboundPackets1_21_2;
|
import com.viaversion.viaversion.protocols.v1_21to1_21_2.packet.ServerboundPackets1_21_2;
|
||||||
import com.viaversion.viaversion.protocols.v1_21to1_21_2.rewriter.RecipeRewriter1_21_2;
|
|
||||||
import com.viaversion.viaversion.rewriter.BlockRewriter;
|
import com.viaversion.viaversion.rewriter.BlockRewriter;
|
||||||
|
import com.viaversion.viaversion.rewriter.RecipeDisplayRewriter;
|
||||||
|
|
||||||
// To replace if needed:
|
// To replace if needed:
|
||||||
// ChunkType1_20_2
|
// ChunkType1_20_2
|
||||||
@ -66,6 +66,9 @@ final class BlockItemPacketRewriter1_99 extends BackwardsStructuredItemRewriter<
|
|||||||
registerContainerClick1_21_2(ServerboundPackets1_21_2.CONTAINER_CLICK);
|
registerContainerClick1_21_2(ServerboundPackets1_21_2.CONTAINER_CLICK);
|
||||||
registerSetCreativeModeSlot(ServerboundPackets1_21_2.SET_CREATIVE_MODE_SLOT);
|
registerSetCreativeModeSlot(ServerboundPackets1_21_2.SET_CREATIVE_MODE_SLOT);
|
||||||
|
|
||||||
new RecipeRewriter1_21_2<>(protocol).register1_20_5(ClientboundPackets1_21_2.UPDATE_RECIPES);
|
final RecipeDisplayRewriter<ClientboundPacket1_21_2> recipeRewriter = new RecipeDisplayRewriter<>(protocol);
|
||||||
|
recipeRewriter.registerUpdateRecipes(ClientboundPackets1_21_2.UPDATE_RECIPES);
|
||||||
|
recipeRewriter.registerRecipeBookAdd(ClientboundPackets1_21_2.RECIPE_BOOK_ADD);
|
||||||
|
recipeRewriter.registerPlaceGhostRecipe(ClientboundPackets1_21_2.PLACE_GHOST_RECIPE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -26,6 +26,7 @@ import com.viaversion.viabackwards.protocol.v1_21_2to1_21.rewriter.ComponentRewr
|
|||||||
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.rewriter.EntityPacketRewriter1_21_2;
|
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.rewriter.EntityPacketRewriter1_21_2;
|
||||||
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.storage.InventoryStateIdStorage;
|
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.storage.InventoryStateIdStorage;
|
||||||
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.storage.ItemTagStorage;
|
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.storage.ItemTagStorage;
|
||||||
|
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.storage.RecipeStorage;
|
||||||
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.storage.SneakingStorage;
|
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.storage.SneakingStorage;
|
||||||
import com.viaversion.viaversion.api.connection.UserConnection;
|
import com.viaversion.viaversion.api.connection.UserConnection;
|
||||||
import com.viaversion.viaversion.api.minecraft.entities.EntityTypes1_20_5;
|
import com.viaversion.viaversion.api.minecraft.entities.EntityTypes1_20_5;
|
||||||
@ -151,6 +152,7 @@ public final class Protocol1_21_2To1_21 extends BackwardsProtocol<ClientboundPac
|
|||||||
user.put(new InventoryStateIdStorage());
|
user.put(new InventoryStateIdStorage());
|
||||||
user.put(new ItemTagStorage());
|
user.put(new ItemTagStorage());
|
||||||
user.put(new SneakingStorage());
|
user.put(new SneakingStorage());
|
||||||
|
user.put(new RecipeStorage(this));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -20,6 +20,7 @@ package com.viaversion.viabackwards.protocol.v1_21_2to1_21.rewriter;
|
|||||||
import com.viaversion.viabackwards.api.rewriters.BackwardsStructuredItemRewriter;
|
import com.viaversion.viabackwards.api.rewriters.BackwardsStructuredItemRewriter;
|
||||||
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.Protocol1_21_2To1_21;
|
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.Protocol1_21_2To1_21;
|
||||||
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.storage.InventoryStateIdStorage;
|
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.storage.InventoryStateIdStorage;
|
||||||
|
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.storage.RecipeStorage;
|
||||||
import com.viaversion.viaversion.api.connection.UserConnection;
|
import com.viaversion.viaversion.api.connection.UserConnection;
|
||||||
import com.viaversion.viaversion.api.data.MappingData;
|
import com.viaversion.viaversion.api.data.MappingData;
|
||||||
import com.viaversion.viaversion.api.minecraft.Particle;
|
import com.viaversion.viaversion.api.minecraft.Particle;
|
||||||
@ -178,33 +179,55 @@ public final class BlockItemPacketRewriter1_21_2 extends BackwardsStructuredItem
|
|||||||
});
|
});
|
||||||
|
|
||||||
protocol.registerClientbound(ClientboundPackets1_21_2.RECIPE_BOOK_ADD, null, wrapper -> {
|
protocol.registerClientbound(ClientboundPackets1_21_2.RECIPE_BOOK_ADD, null, wrapper -> {
|
||||||
// TODO
|
final RecipeStorage recipeStorage = wrapper.user().get(RecipeStorage.class);
|
||||||
|
recipeStorage.clearRecipes();
|
||||||
|
|
||||||
|
final int size = wrapper.read(Types.VAR_INT);
|
||||||
|
for (int i = 0; i < size; i++) {
|
||||||
|
recipeStorage.readRecipe(wrapper);
|
||||||
|
}
|
||||||
|
recipeStorage.sendRecipes(wrapper.user());
|
||||||
wrapper.cancel();
|
wrapper.cancel();
|
||||||
});
|
});
|
||||||
protocol.registerClientbound(ClientboundPackets1_21_2.RECIPE_BOOK_REMOVE, null, wrapper -> {
|
protocol.registerClientbound(ClientboundPackets1_21_2.RECIPE_BOOK_REMOVE, ClientboundPackets1_21.RECIPE, wrapper -> {
|
||||||
// TODO
|
final RecipeStorage recipeStorage = wrapper.user().get(RecipeStorage.class);
|
||||||
wrapper.cancel();
|
final int[] ids = wrapper.read(Types.VAR_INT_ARRAY_PRIMITIVE);
|
||||||
|
recipeStorage.lockRecipes(wrapper, ids);
|
||||||
});
|
});
|
||||||
protocol.registerClientbound(ClientboundPackets1_21_2.RECIPE_BOOK_SETTINGS, null, wrapper -> {
|
protocol.registerClientbound(ClientboundPackets1_21_2.RECIPE_BOOK_SETTINGS, null, wrapper -> {
|
||||||
// TODO
|
final RecipeStorage recipeStorage = wrapper.user().get(RecipeStorage.class);
|
||||||
|
final boolean[] settings = new boolean[RecipeStorage.RECIPE_BOOK_SETTINGS];
|
||||||
|
for (int i = 0; i < RecipeStorage.RECIPE_BOOK_SETTINGS; i++) {
|
||||||
|
settings[i] = wrapper.read(Types.BOOLEAN);
|
||||||
|
}
|
||||||
|
recipeStorage.setRecipeBookSettings(settings);
|
||||||
|
|
||||||
wrapper.cancel();
|
wrapper.cancel();
|
||||||
});
|
});
|
||||||
protocol.registerClientbound(ClientboundPackets1_21_2.UPDATE_RECIPES, wrapper -> {
|
protocol.registerClientbound(ClientboundPackets1_21_2.UPDATE_RECIPES, wrapper -> {
|
||||||
wrapper.cancel(); // TODO
|
// Inputs for furnaces etc., hardcoded in earlier versions
|
||||||
|
final int size = wrapper.passthrough(Types.VAR_INT);
|
||||||
|
for (int i = 0; i < size; i++) {
|
||||||
|
wrapper.read(Types.STRING); // Recipe group
|
||||||
|
wrapper.read(Types.VAR_INT_ARRAY_PRIMITIVE); // Items
|
||||||
|
}
|
||||||
|
|
||||||
|
final RecipeStorage recipeStorage = wrapper.user().get(RecipeStorage.class);
|
||||||
|
recipeStorage.readStoneCutterRecipes(wrapper);
|
||||||
|
|
||||||
|
// Send later with the recipe book init
|
||||||
|
wrapper.cancel();
|
||||||
});
|
});
|
||||||
protocol.registerClientbound(ClientboundPackets1_21_2.PLACE_GHOST_RECIPE, wrapper -> {
|
protocol.registerClientbound(ClientboundPackets1_21_2.PLACE_GHOST_RECIPE, wrapper -> {
|
||||||
this.updateContainerId(wrapper);
|
this.updateContainerId(wrapper);
|
||||||
final int recipeDisplay = wrapper.read(Types.VAR_INT);
|
|
||||||
wrapper.cancel(); // TODO
|
wrapper.cancel(); // TODO
|
||||||
});
|
});
|
||||||
protocol.registerServerbound(ServerboundPackets1_20_5.PLACE_RECIPE, wrapper -> {
|
protocol.registerServerbound(ServerboundPackets1_20_5.PLACE_RECIPE, wrapper -> {
|
||||||
this.updateContainerIdServerbound(wrapper);
|
this.updateContainerIdServerbound(wrapper);
|
||||||
final int recipeDisplayId = wrapper.read(Types.VAR_INT);
|
|
||||||
wrapper.cancel(); // TODO
|
wrapper.cancel(); // TODO
|
||||||
});
|
});
|
||||||
protocol.registerServerbound(ServerboundPackets1_20_5.RECIPE_BOOK_SEEN_RECIPE, wrapper -> {
|
protocol.registerServerbound(ServerboundPackets1_20_5.RECIPE_BOOK_SEEN_RECIPE, wrapper -> {
|
||||||
this.updateContainerIdServerbound(wrapper);
|
this.updateContainerIdServerbound(wrapper);
|
||||||
final int recipeDisplayId = wrapper.read(Types.VAR_INT);
|
|
||||||
wrapper.cancel(); // TODO
|
wrapper.cancel(); // TODO
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,406 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ViaBackwards - https://github.com/ViaVersion/ViaBackwards
|
||||||
|
* Copyright (C) 2016-2024 ViaVersion and contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
package com.viaversion.viabackwards.protocol.v1_21_2to1_21.storage;
|
||||||
|
|
||||||
|
import com.google.common.base.Preconditions;
|
||||||
|
import com.viaversion.viabackwards.protocol.v1_21_2to1_21.Protocol1_21_2To1_21;
|
||||||
|
import com.viaversion.viaversion.api.connection.StorableObject;
|
||||||
|
import com.viaversion.viaversion.api.connection.UserConnection;
|
||||||
|
import com.viaversion.viaversion.api.minecraft.HolderSet;
|
||||||
|
import com.viaversion.viaversion.api.minecraft.item.Item;
|
||||||
|
import com.viaversion.viaversion.api.minecraft.item.StructuredItem;
|
||||||
|
import com.viaversion.viaversion.api.protocol.packet.PacketWrapper;
|
||||||
|
import com.viaversion.viaversion.api.type.Types;
|
||||||
|
import com.viaversion.viaversion.api.type.types.version.Types1_21_2;
|
||||||
|
import com.viaversion.viaversion.protocols.v1_20_5to1_21.packet.ClientboundPackets1_21;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public final class RecipeStorage implements StorableObject {
|
||||||
|
|
||||||
|
// Pairs of open + filtering for: Crafting, furnace, blast furnace, smoker
|
||||||
|
public static final int RECIPE_BOOK_SETTINGS = 4 * 2;
|
||||||
|
private final List<Recipe> recipes = new ArrayList<>();
|
||||||
|
private final List<StoneCutterRecipe> stoneCutterRecipes = new ArrayList<>();
|
||||||
|
private boolean[] recipeBookSettings = new boolean[RECIPE_BOOK_SETTINGS];
|
||||||
|
private final Protocol1_21_2To1_21 protocol;
|
||||||
|
private int highestIndex;
|
||||||
|
|
||||||
|
public RecipeStorage(final Protocol1_21_2To1_21 protocol) {
|
||||||
|
this.protocol = protocol;
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract static class Recipe {
|
||||||
|
protected int index;
|
||||||
|
private Integer group;
|
||||||
|
private int category;
|
||||||
|
private boolean locked;
|
||||||
|
|
||||||
|
abstract void write(PacketWrapper wrapper);
|
||||||
|
|
||||||
|
void writeGroup(final PacketWrapper wrapper) {
|
||||||
|
wrapper.write(Types.STRING, group != null ? Integer.toString(group) : "");
|
||||||
|
}
|
||||||
|
|
||||||
|
void writeIngredients(final PacketWrapper wrapper, final Item[][] ingredients) {
|
||||||
|
wrapper.write(Types.VAR_INT, ingredients.length);
|
||||||
|
for (final Item[] ingredient : ingredients) {
|
||||||
|
writeIngredient(wrapper, ingredient);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void writeIngredient(final PacketWrapper wrapper, final Item[] ingredient) {
|
||||||
|
final Item[] copy = new Item[ingredient.length];
|
||||||
|
for (int i = 0; i < ingredient.length; i++) {
|
||||||
|
copy[i] = ingredient[i].copy();
|
||||||
|
}
|
||||||
|
wrapper.write(Types1_21_2.ITEM_ARRAY, copy);
|
||||||
|
}
|
||||||
|
|
||||||
|
void writeResult(final PacketWrapper wrapper, final Item result) {
|
||||||
|
wrapper.write(Types1_21_2.ITEM, result.copy());
|
||||||
|
}
|
||||||
|
|
||||||
|
void writeCategory(final PacketWrapper wrapper) {
|
||||||
|
wrapper.write(Types.VAR_INT, 0); // TODO
|
||||||
|
}
|
||||||
|
|
||||||
|
int category() {
|
||||||
|
return category;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void sendRecipes(final UserConnection connection) {
|
||||||
|
// Add stonecutter recipes from update_recipes
|
||||||
|
final List<Recipe> recipes = new ArrayList<>(this.recipes);
|
||||||
|
for (final StoneCutterRecipe recipe : stoneCutterRecipes) {
|
||||||
|
recipe.index = ++highestIndex;
|
||||||
|
recipes.add(recipe);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Since the server only sends unlocked recipes, we need to re-send all recipes in UPDATE_RECIPES
|
||||||
|
final PacketWrapper updateRecipesPacket = PacketWrapper.create(ClientboundPackets1_21.UPDATE_RECIPES, connection);
|
||||||
|
updateRecipesPacket.write(Types.VAR_INT, recipes.size());
|
||||||
|
for (final Recipe recipe : recipes) {
|
||||||
|
updateRecipesPacket.write(Types.STRING, Integer.toString(recipe.index)); // Use index as the recipe identifier
|
||||||
|
recipe.write(updateRecipesPacket);
|
||||||
|
}
|
||||||
|
updateRecipesPacket.send(Protocol1_21_2To1_21.class);
|
||||||
|
|
||||||
|
sendUnlockedRecipes(connection, recipes);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void lockRecipes(final PacketWrapper wrapper, final int[] ids) {
|
||||||
|
for (final int id : ids) {
|
||||||
|
recipes.get(id).locked = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
wrapper.write(Types.VAR_INT, 2); // Remove recipes
|
||||||
|
for (final boolean recipeBookSetting : recipeBookSettings) {
|
||||||
|
wrapper.write(Types.BOOLEAN, recipeBookSetting);
|
||||||
|
}
|
||||||
|
|
||||||
|
final String[] recipeKeys = new String[ids.length];
|
||||||
|
for (int i = 0; i < ids.length; i++) {
|
||||||
|
recipeKeys[i] = Integer.toString(ids[i]);
|
||||||
|
}
|
||||||
|
wrapper.write(Types.STRING_ARRAY, recipeKeys);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void sendUnlockedRecipes(final UserConnection connection, final List<Recipe> recipes) {
|
||||||
|
// TODO Not working?
|
||||||
|
final PacketWrapper wrapper = PacketWrapper.create(ClientboundPackets1_21.RECIPE, connection);
|
||||||
|
wrapper.write(Types.VAR_INT, 0); // Init recipes
|
||||||
|
|
||||||
|
for (final boolean recipeBookSetting : recipeBookSettings) {
|
||||||
|
wrapper.write(Types.BOOLEAN, recipeBookSetting);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use index as the recipe identifier. We only know the unlocked ones, so send all
|
||||||
|
final String[] recipeKeys = new String[recipes.size()];
|
||||||
|
for (int i = 0; i < recipes.size(); i++) {
|
||||||
|
recipeKeys[i] = Integer.toString(i);
|
||||||
|
}
|
||||||
|
wrapper.write(Types.STRING_ARRAY, recipeKeys);
|
||||||
|
|
||||||
|
wrapper.write(Types.STRING_ARRAY, new String[0]); // Highlights // TODO
|
||||||
|
wrapper.send(Protocol1_21_2To1_21.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void readRecipe(final PacketWrapper wrapper) {
|
||||||
|
final int id = wrapper.read(Types.VAR_INT);
|
||||||
|
final int type = wrapper.passthrough(Types.VAR_INT);
|
||||||
|
final Recipe recipe = switch (type) {
|
||||||
|
case 0 -> readShapeless(wrapper);
|
||||||
|
case 1 -> readShaped(wrapper);
|
||||||
|
case 2 -> readFurnace(wrapper);
|
||||||
|
case 3 -> readStoneCutter(wrapper);
|
||||||
|
case 4 -> readSmithing(wrapper);
|
||||||
|
default -> null;
|
||||||
|
};
|
||||||
|
|
||||||
|
final Integer group = wrapper.read(Types.OPTIONAL_VAR_INT);
|
||||||
|
final int category = wrapper.read(Types.VAR_INT);
|
||||||
|
if (wrapper.read(Types.BOOLEAN)) {
|
||||||
|
final int ingredientsSize = wrapper.read(Types.VAR_INT);
|
||||||
|
for (int j = 0; j < ingredientsSize; j++) {
|
||||||
|
//handleIngredient(wrapper); // Items //TODO
|
||||||
|
wrapper.read(Types.HOLDER_SET);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
final byte flags = wrapper.read(Types.BYTE);
|
||||||
|
|
||||||
|
if (recipe != null) {
|
||||||
|
recipe.index = id;
|
||||||
|
recipe.group = group;
|
||||||
|
recipe.category = category;
|
||||||
|
}
|
||||||
|
highestIndex = Math.max(highestIndex, id);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Recipe readShapeless(final PacketWrapper wrapper) {
|
||||||
|
final Item[][] ingredients = readSlotDisplayList(wrapper);
|
||||||
|
final Item result = readSingleSlotDisplay(wrapper);
|
||||||
|
readSlotDisplay(wrapper); // Crafting station
|
||||||
|
return add(new ShapelessRecipe(ingredients, result));
|
||||||
|
}
|
||||||
|
|
||||||
|
private Recipe readShaped(final PacketWrapper wrapper) {
|
||||||
|
final int width = wrapper.passthrough(Types.VAR_INT);
|
||||||
|
final int height = wrapper.passthrough(Types.VAR_INT);
|
||||||
|
final Item[][] ingredients = readSlotDisplayList(wrapper);
|
||||||
|
final Item result = readSingleSlotDisplay(wrapper);
|
||||||
|
readSlotDisplay(wrapper); // Crafting station
|
||||||
|
return add(new ShapedRecipe(width, height, ingredients, result));
|
||||||
|
}
|
||||||
|
|
||||||
|
private Recipe readFurnace(final PacketWrapper wrapper) {
|
||||||
|
final Item[] ingredient = readSlotDisplay(wrapper);
|
||||||
|
readSlotDisplay(wrapper); // Fuel
|
||||||
|
final Item result = readSingleSlotDisplay(wrapper);
|
||||||
|
readSlotDisplay(wrapper); // Crafting station
|
||||||
|
return add(new FurnaceRecipe(ingredient, result));
|
||||||
|
}
|
||||||
|
|
||||||
|
private Recipe readStoneCutter(final PacketWrapper wrapper) {
|
||||||
|
// Use values from UPDATE_RECIPES instead
|
||||||
|
readSlotDisplay(wrapper); // Result
|
||||||
|
readSlotDisplay(wrapper); // Crafting station
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Recipe readSmithing(final PacketWrapper wrapper) {
|
||||||
|
// TODO Combine with update_recipes
|
||||||
|
readSlotDisplay(wrapper); // Result
|
||||||
|
readSlotDisplay(wrapper); // Crafting station
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Recipe add(final Recipe recipe) {
|
||||||
|
recipes.add(recipe);
|
||||||
|
return recipe;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Item[][] readSlotDisplayList(final PacketWrapper wrapper) {
|
||||||
|
final int size = wrapper.passthrough(Types.VAR_INT);
|
||||||
|
final Item[][] ingredients = new Item[size][];
|
||||||
|
for (int i = 0; i < size; i++) {
|
||||||
|
ingredients[i] = readSlotDisplay(wrapper);
|
||||||
|
}
|
||||||
|
return ingredients;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Item readSingleSlotDisplay(final PacketWrapper wrapper) {
|
||||||
|
final Item[] items = readSlotDisplay(wrapper);
|
||||||
|
return items.length == 0 ? new StructuredItem(1, 1) : items[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
private Item[] readSlotDisplay(final PacketWrapper wrapper) {
|
||||||
|
// empty, any_fuel, smithing_trim are empty
|
||||||
|
final int type = wrapper.read(Types.VAR_INT);
|
||||||
|
return switch (type) {
|
||||||
|
case 2 -> {
|
||||||
|
final int id = wrapper.read(Types.VAR_INT);
|
||||||
|
if (id == 0) {
|
||||||
|
protocol.getLogger().warning("Empty item id in recipe");
|
||||||
|
yield new Item[0];
|
||||||
|
}
|
||||||
|
yield new Item[]{new StructuredItem(rewriteItemId(id), 1)};
|
||||||
|
}
|
||||||
|
case 3 -> {
|
||||||
|
final Item item = wrapper.read(Types1_21_2.ITEM);
|
||||||
|
protocol.getItemRewriter().handleItemToClient(wrapper.user(), item);
|
||||||
|
if (item.isEmpty()) {
|
||||||
|
protocol.getLogger().warning("Empty item in recipe");
|
||||||
|
yield new Item[0];
|
||||||
|
}
|
||||||
|
yield new Item[]{item};
|
||||||
|
}
|
||||||
|
case 4 -> {
|
||||||
|
wrapper.read(Types.STRING); // Tag key // TODO
|
||||||
|
yield new Item[0];
|
||||||
|
}
|
||||||
|
case 6 -> readSlotDisplayList(wrapper)[0]; // Composite
|
||||||
|
default -> new Item[0];
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private int rewriteItemId(final int id) {
|
||||||
|
return protocol.getMappingData().getNewItemId(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void readStoneCutterRecipes(final PacketWrapper wrapper) {
|
||||||
|
stoneCutterRecipes.clear();
|
||||||
|
final int stonecutterRecipesSize = wrapper.read(Types.VAR_INT);
|
||||||
|
for (int i = 0; i < stonecutterRecipesSize; i++) {
|
||||||
|
// The ingredients are what's actually used in client prediction, they're the important part
|
||||||
|
final Item[] ingredient = readHolderSet(wrapper);
|
||||||
|
// TODO Probably not actually the result, might have to combine with update_recipes
|
||||||
|
final Item result = readSingleSlotDisplay(wrapper);
|
||||||
|
stoneCutterRecipes.add(new StoneCutterRecipe(ingredient, result));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private Item[] readHolderSet(final PacketWrapper wrapper) {
|
||||||
|
final HolderSet holderSet = wrapper.read(Types.HOLDER_SET);
|
||||||
|
if (holderSet.hasTagKey()) {
|
||||||
|
return new Item[]{new StructuredItem(1, 1)}; // TODO
|
||||||
|
}
|
||||||
|
|
||||||
|
final int[] ids = holderSet.ids();
|
||||||
|
for (int i = 0; i < ids.length; i++) {
|
||||||
|
ids[i] = rewriteItemId(ids[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
final Item[] ingredient = new Item[ids.length];
|
||||||
|
for (int i = 0; i < ingredient.length; i++) {
|
||||||
|
ingredient[i] = new StructuredItem(ids[i], 1);
|
||||||
|
}
|
||||||
|
return ingredient;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final class ShapelessRecipe extends Recipe {
|
||||||
|
private static final int SERIALIZER_ID = 1;
|
||||||
|
private final Item[][] ingredients;
|
||||||
|
private final Item result;
|
||||||
|
|
||||||
|
private ShapelessRecipe(final Item[][] ingredients, final Item result) {
|
||||||
|
this.ingredients = ingredients;
|
||||||
|
this.result = result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(final PacketWrapper wrapper) {
|
||||||
|
wrapper.write(Types.VAR_INT, SERIALIZER_ID);
|
||||||
|
writeGroup(wrapper);
|
||||||
|
writeCategory(wrapper);
|
||||||
|
writeIngredients(wrapper, ingredients);
|
||||||
|
writeResult(wrapper, result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final class ShapedRecipe extends Recipe {
|
||||||
|
private static final int SERIALIZER_ID = 0;
|
||||||
|
private final int width;
|
||||||
|
private final int height;
|
||||||
|
private final Item[][] ingredients;
|
||||||
|
private final Item result;
|
||||||
|
|
||||||
|
private ShapedRecipe(final int width, final int height, final Item[][] ingredients, final Item result) {
|
||||||
|
this.width = width;
|
||||||
|
this.height = height;
|
||||||
|
this.ingredients = ingredients;
|
||||||
|
this.result = result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(final PacketWrapper wrapper) {
|
||||||
|
wrapper.write(Types.VAR_INT, SERIALIZER_ID);
|
||||||
|
writeGroup(wrapper);
|
||||||
|
writeCategory(wrapper);
|
||||||
|
wrapper.write(Types.VAR_INT, width);
|
||||||
|
wrapper.write(Types.VAR_INT, height);
|
||||||
|
Preconditions.checkArgument(width * height == ingredients.length, "Invalid shaped recipe");
|
||||||
|
// No length prefix
|
||||||
|
for (final Item[] ingredient : ingredients) {
|
||||||
|
writeIngredient(wrapper, ingredient);
|
||||||
|
}
|
||||||
|
writeResult(wrapper, result);
|
||||||
|
wrapper.write(Types.BOOLEAN, false); // Doesn't matter for the init
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final class FurnaceRecipe extends Recipe {
|
||||||
|
private final Item[] ingredient;
|
||||||
|
private final Item result;
|
||||||
|
|
||||||
|
private FurnaceRecipe(final Item[] ingredient, final Item result) {
|
||||||
|
this.ingredient = ingredient;
|
||||||
|
this.result = result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(final PacketWrapper wrapper) {
|
||||||
|
wrapper.write(Types.VAR_INT, serializerId());
|
||||||
|
writeGroup(wrapper);
|
||||||
|
writeCategory(wrapper);
|
||||||
|
writeIngredient(wrapper, ingredient);
|
||||||
|
writeResult(wrapper, result);
|
||||||
|
wrapper.write(Types.FLOAT, 0F); // XP
|
||||||
|
wrapper.write(Types.VAR_INT, 200); // Cooking time, determined by the client in 1.21.2
|
||||||
|
}
|
||||||
|
|
||||||
|
private int serializerId() {
|
||||||
|
return switch (category()) {
|
||||||
|
case 4, 5, 6 -> 15; // Furnace food, bocks, misc
|
||||||
|
case 7, 8 -> 16; // Blast furnace blocks, misc
|
||||||
|
case 9 -> 17; // Smoker
|
||||||
|
case 12 -> 18; // Campfire
|
||||||
|
default -> 15;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final class StoneCutterRecipe extends Recipe {
|
||||||
|
private static final int SERIALIZER_ID = 19;
|
||||||
|
private final Item[] ingredient;
|
||||||
|
private final Item result;
|
||||||
|
|
||||||
|
private StoneCutterRecipe(final Item[] ingredient, final Item result) {
|
||||||
|
this.ingredient = ingredient;
|
||||||
|
this.result = result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(final PacketWrapper wrapper) {
|
||||||
|
wrapper.write(Types.VAR_INT, SERIALIZER_ID);
|
||||||
|
writeGroup(wrapper);
|
||||||
|
writeIngredient(wrapper, ingredient);
|
||||||
|
writeResult(wrapper, result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setRecipeBookSettings(final boolean[] recipeBookSettings) {
|
||||||
|
this.recipeBookSettings = recipeBookSettings;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void clearRecipes() {
|
||||||
|
recipes.clear();
|
||||||
|
highestIndex = 0;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user