mirror of
https://github.com/FabricMC/fabric.git
synced 2025-04-21 03:10:54 -04:00
[1.20.5] Add back custom data ingredient (#3642)
* [1.20.5] Add back custom data ingredient --------- Co-authored-by: modmuss50 <modmuss50@gmail.com>
This commit is contained in:
parent
168bf74f8b
commit
f628d01844
4 changed files with 218 additions and 0 deletions
fabric-recipe-api-v1/src
main/java/net/fabricmc/fabric
api/recipe/v1/ingredient
impl/recipe/ingredient
testmod/java/net/fabricmc/fabric/test/recipe/ingredient
|
@ -22,11 +22,14 @@ import java.util.function.UnaryOperator;
|
|||
|
||||
import net.minecraft.component.ComponentChanges;
|
||||
import net.minecraft.item.ItemStack;
|
||||
import net.minecraft.nbt.NbtCompound;
|
||||
import net.minecraft.nbt.NbtHelper;
|
||||
import net.minecraft.recipe.Ingredient;
|
||||
|
||||
import net.fabricmc.fabric.impl.recipe.ingredient.builtin.AllIngredient;
|
||||
import net.fabricmc.fabric.impl.recipe.ingredient.builtin.AnyIngredient;
|
||||
import net.fabricmc.fabric.impl.recipe.ingredient.builtin.ComponentsIngredient;
|
||||
import net.fabricmc.fabric.impl.recipe.ingredient.builtin.CustomDataIngredient;
|
||||
import net.fabricmc.fabric.impl.recipe.ingredient.builtin.DifferenceIngredient;
|
||||
|
||||
/**
|
||||
|
@ -151,6 +154,39 @@ public final class DefaultCustomIngredients {
|
|||
return components(Ingredient.ofItems(stack.getItem()), stack.getComponentChanges());
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates an ingredient that wraps another ingredient to also check for stack's {@linkplain
|
||||
* net.minecraft.component.DataComponentTypes#CUSTOM_DATA custom data}.
|
||||
* This check is non-strict; the ingredient custom data must be a subset of the stack custom data.
|
||||
* This is useful for mods that still rely on NBT-based custom data instead of custom components,
|
||||
* such as those requiring vanilla compatibility or interacting with another data packs.
|
||||
*
|
||||
* <p>Passing a {@code null} or empty {@code nbt} is <strong>not</strong> allowed, as it would always match.
|
||||
* For strict matching, use {@link #components(Ingredient, UnaryOperator)} like this instead:
|
||||
*
|
||||
* <pre>{@code
|
||||
* components(base, builder -> builder.add(DataComponentTypes.CUSTOM_DATA, NbtComponent.of(nbt)));
|
||||
* // or, to check for absence of custom data:
|
||||
* components(base, builder -> builder.remove(DataComponentTypes.CUSTOM_DATA));
|
||||
* }</pre>
|
||||
*
|
||||
* <p>See {@link NbtHelper#matches} for how matching works.
|
||||
*
|
||||
* <p>The JSON format is as follows:
|
||||
* <pre>{@code
|
||||
* {
|
||||
* "fabric:type": "fabric:custom_data",
|
||||
* "base": // base ingredient,
|
||||
* "nbt": // NBT tag to match, either in JSON directly or a string representation
|
||||
* }
|
||||
* }</pre>
|
||||
*
|
||||
* @throws IllegalArgumentException if {@code nbt} is {@code null} or empty
|
||||
*/
|
||||
public static Ingredient customData(Ingredient base, NbtCompound nbt) {
|
||||
return new CustomDataIngredient(base, nbt).toVanilla();
|
||||
}
|
||||
|
||||
private DefaultCustomIngredients() {
|
||||
}
|
||||
}
|
||||
|
|
|
@ -21,6 +21,7 @@ import net.fabricmc.fabric.api.recipe.v1.ingredient.CustomIngredientSerializer;
|
|||
import net.fabricmc.fabric.impl.recipe.ingredient.builtin.AllIngredient;
|
||||
import net.fabricmc.fabric.impl.recipe.ingredient.builtin.AnyIngredient;
|
||||
import net.fabricmc.fabric.impl.recipe.ingredient.builtin.ComponentsIngredient;
|
||||
import net.fabricmc.fabric.impl.recipe.ingredient.builtin.CustomDataIngredient;
|
||||
import net.fabricmc.fabric.impl.recipe.ingredient.builtin.DifferenceIngredient;
|
||||
|
||||
/**
|
||||
|
@ -33,5 +34,6 @@ public class CustomIngredientInit implements ModInitializer {
|
|||
CustomIngredientSerializer.register(AnyIngredient.SERIALIZER);
|
||||
CustomIngredientSerializer.register(DifferenceIngredient.SERIALIZER);
|
||||
CustomIngredientSerializer.register(ComponentsIngredient.SERIALIZER);
|
||||
CustomIngredientSerializer.register(CustomDataIngredient.SERIALIZER);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -0,0 +1,141 @@
|
|||
/*
|
||||
* Copyright (c) 2016, 2017, 2018, 2019 FabricMC
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package net.fabricmc.fabric.impl.recipe.ingredient.builtin;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
import com.mojang.brigadier.exceptions.CommandSyntaxException;
|
||||
import com.mojang.datafixers.util.Either;
|
||||
import com.mojang.serialization.Codec;
|
||||
import com.mojang.serialization.DataResult;
|
||||
import com.mojang.serialization.codecs.RecordCodecBuilder;
|
||||
|
||||
import net.minecraft.component.DataComponentTypes;
|
||||
import net.minecraft.component.type.NbtComponent;
|
||||
import net.minecraft.item.ItemStack;
|
||||
import net.minecraft.nbt.NbtCompound;
|
||||
import net.minecraft.nbt.StringNbtReader;
|
||||
import net.minecraft.network.RegistryByteBuf;
|
||||
import net.minecraft.network.codec.PacketCodec;
|
||||
import net.minecraft.network.codec.PacketCodecs;
|
||||
import net.minecraft.recipe.Ingredient;
|
||||
import net.minecraft.util.Identifier;
|
||||
import net.minecraft.util.dynamic.Codecs;
|
||||
|
||||
import net.fabricmc.fabric.api.recipe.v1.ingredient.CustomIngredient;
|
||||
import net.fabricmc.fabric.api.recipe.v1.ingredient.CustomIngredientSerializer;
|
||||
|
||||
public class CustomDataIngredient implements CustomIngredient {
|
||||
public static final CustomIngredientSerializer<CustomDataIngredient> SERIALIZER = new Serializer();
|
||||
private final Ingredient base;
|
||||
private final NbtCompound nbt;
|
||||
|
||||
public CustomDataIngredient(Ingredient base, NbtCompound nbt) {
|
||||
if (nbt == null || nbt.isEmpty()) throw new IllegalArgumentException("NBT cannot be null; use components ingredient for strict matching");
|
||||
|
||||
this.base = base;
|
||||
this.nbt = nbt;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean test(ItemStack stack) {
|
||||
if (!base.test(stack)) return false;
|
||||
|
||||
NbtComponent nbt = stack.get(DataComponentTypes.CUSTOM_DATA);
|
||||
|
||||
return nbt != null && nbt.matches(this.nbt);
|
||||
}
|
||||
|
||||
@Override
|
||||
public List<ItemStack> getMatchingStacks() {
|
||||
List<ItemStack> stacks = new ArrayList<>(List.of(base.getMatchingStacks()));
|
||||
stacks.replaceAll(stack -> {
|
||||
ItemStack copy = stack.copy();
|
||||
copy.apply(DataComponentTypes.CUSTOM_DATA, NbtComponent.DEFAULT, existingNbt -> NbtComponent.of(existingNbt.copyNbt().copyFrom(this.nbt)));
|
||||
return copy;
|
||||
});
|
||||
stacks.removeIf(stack -> !base.test(stack));
|
||||
return stacks;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean requiresTesting() {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public CustomIngredientSerializer<?> getSerializer() {
|
||||
return SERIALIZER;
|
||||
}
|
||||
|
||||
private Ingredient getBase() {
|
||||
return base;
|
||||
}
|
||||
|
||||
private NbtCompound getNbt() {
|
||||
return nbt;
|
||||
}
|
||||
|
||||
private static class Serializer implements CustomIngredientSerializer<CustomDataIngredient> {
|
||||
private static final Identifier ID = new Identifier("fabric", "custom_data");
|
||||
|
||||
// Supports decoding the NBT as a string as well as the object.
|
||||
private static final Codec<NbtCompound> NBT_CODEC = Codecs.xor(
|
||||
Codec.STRING, NbtCompound.CODEC
|
||||
).flatXmap(either -> either.map(s -> {
|
||||
try {
|
||||
return DataResult.success(StringNbtReader.parse(s));
|
||||
} catch (CommandSyntaxException e) {
|
||||
return DataResult.error(e::getMessage);
|
||||
}
|
||||
}, DataResult::success), nbtCompound -> DataResult.success(Either.left(nbtCompound.asString())));
|
||||
|
||||
private static final Codec<CustomDataIngredient> ALLOW_EMPTY_CODEC = createCodec(Ingredient.ALLOW_EMPTY_CODEC);
|
||||
private static final Codec<CustomDataIngredient> DISALLOW_EMPTY_CODEC = createCodec(Ingredient.DISALLOW_EMPTY_CODEC);
|
||||
|
||||
private static final PacketCodec<RegistryByteBuf, CustomDataIngredient> PACKET_CODEC = PacketCodec.tuple(
|
||||
Ingredient.PACKET_CODEC, CustomDataIngredient::getBase,
|
||||
PacketCodecs.NBT_COMPOUND, CustomDataIngredient::getNbt,
|
||||
CustomDataIngredient::new
|
||||
);
|
||||
|
||||
private static Codec<CustomDataIngredient> createCodec(Codec<Ingredient> ingredientCodec) {
|
||||
return RecordCodecBuilder.create(instance ->
|
||||
instance.group(
|
||||
ingredientCodec.fieldOf("base").forGetter(CustomDataIngredient::getBase),
|
||||
NBT_CODEC.fieldOf("nbt").forGetter(CustomDataIngredient::getNbt)
|
||||
).apply(instance, CustomDataIngredient::new)
|
||||
);
|
||||
}
|
||||
|
||||
@Override
|
||||
public Identifier getIdentifier() {
|
||||
return ID;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Codec<CustomDataIngredient> getCodec(boolean allowEmpty) {
|
||||
return allowEmpty ? ALLOW_EMPTY_CODEC : DISALLOW_EMPTY_CODEC;
|
||||
}
|
||||
|
||||
@Override
|
||||
public PacketCodec<RegistryByteBuf, CustomDataIngredient> getPacketCodec() {
|
||||
return PACKET_CODEC;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -29,6 +29,7 @@ import net.minecraft.test.GameTest;
|
|||
import net.minecraft.test.GameTestException;
|
||||
import net.minecraft.test.TestContext;
|
||||
import net.minecraft.text.Text;
|
||||
import net.minecraft.util.Util;
|
||||
|
||||
import net.fabricmc.fabric.api.gametest.v1.FabricGameTest;
|
||||
import net.fabricmc.fabric.api.recipe.v1.ingredient.DefaultCustomIngredients;
|
||||
|
@ -158,6 +159,44 @@ public class IngredientMatchTests {
|
|||
context.complete();
|
||||
}
|
||||
|
||||
@GameTest(templateName = FabricGameTest.EMPTY_STRUCTURE)
|
||||
public void testCustomDataIngredient(TestContext context) {
|
||||
final NbtCompound requiredNbt = Util.make(new NbtCompound(), nbt -> {
|
||||
nbt.putInt("keyA", 1);
|
||||
});
|
||||
final NbtCompound acceptedNbt = Util.make(requiredNbt.copy(), nbt -> {
|
||||
nbt.putInt("keyB", 2);
|
||||
});
|
||||
final NbtCompound rejectedNbt1 = Util.make(new NbtCompound(), nbt -> {
|
||||
nbt.putInt("keyA", -1);
|
||||
});
|
||||
final NbtCompound rejectedNbt2 = Util.make(new NbtCompound(), nbt -> {
|
||||
nbt.putInt("keyB", 2);
|
||||
});
|
||||
|
||||
final Ingredient baseIngredient = Ingredient.ofItems(Items.STICK);
|
||||
final Ingredient customDataIngredient = DefaultCustomIngredients.customData(baseIngredient, requiredNbt);
|
||||
|
||||
ItemStack stack = new ItemStack(Items.STICK);
|
||||
assertEquals(false, customDataIngredient.test(stack));
|
||||
stack.set(DataComponentTypes.CUSTOM_DATA, NbtComponent.of(requiredNbt));
|
||||
assertEquals(true, customDataIngredient.test(stack));
|
||||
// This is a non-strict matching
|
||||
stack.set(DataComponentTypes.CUSTOM_DATA, NbtComponent.of(acceptedNbt));
|
||||
assertEquals(true, customDataIngredient.test(stack));
|
||||
stack.set(DataComponentTypes.CUSTOM_DATA, NbtComponent.of(rejectedNbt1));
|
||||
assertEquals(false, customDataIngredient.test(stack));
|
||||
stack.set(DataComponentTypes.CUSTOM_DATA, NbtComponent.of(rejectedNbt2));
|
||||
assertEquals(false, customDataIngredient.test(stack));
|
||||
|
||||
ItemStack[] matchingStacks = customDataIngredient.getMatchingStacks();
|
||||
assertEquals(1, matchingStacks.length);
|
||||
assertEquals(Items.STICK, matchingStacks[0].getItem());
|
||||
assertEquals(NbtComponent.of(requiredNbt), matchingStacks[0].get(DataComponentTypes.CUSTOM_DATA));
|
||||
|
||||
context.complete();
|
||||
}
|
||||
|
||||
private static <T> void assertEquals(T expected, T actual) {
|
||||
if (!Objects.equals(expected, actual)) {
|
||||
throw new GameTestException(String.format("assertEquals failed%nexpected: %s%n but was: %s", expected, actual));
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue