• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

CyclopsMC / IntegratedDynamics / 14811858327

03 May 2025 02:39PM UTC coverage: 45.566% (+0.2%) from 45.387%
14811858327

push

github

rubensworks
Merge remote-tracking branch 'origin/master-1.21-lts' into master-1.21

2559 of 8405 branches covered (30.45%)

Branch coverage included in aggregate %.

11881 of 23285 relevant lines covered (51.02%)

2.43 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

87.03
/src/main/java/org/cyclops/integrateddynamics/core/evaluate/OperatorBuilders.java
1
package org.cyclops.integrateddynamics.core.evaluate;
2

3
import com.google.common.collect.Iterables;
4
import com.google.common.collect.Lists;
5
import net.minecraft.nbt.CompoundTag;
6
import net.minecraft.nbt.Tag;
7
import net.minecraft.network.chat.Component;
8
import net.minecraft.network.chat.MutableComponent;
9
import net.minecraft.resources.ResourceLocation;
10
import net.minecraft.world.entity.Entity;
11
import net.minecraft.world.item.ItemStack;
12
import net.minecraft.world.level.block.SoundType;
13
import net.neoforged.fml.ModContainer;
14
import net.neoforged.fml.ModList;
15
import net.neoforged.neoforge.capabilities.Capabilities;
16
import net.neoforged.neoforge.capabilities.ItemCapability;
17
import net.neoforged.neoforge.energy.IEnergyStorage;
18
import net.neoforged.neoforge.fluids.FluidStack;
19
import org.apache.commons.lang3.ArrayUtils;
20
import org.apache.commons.lang3.tuple.Pair;
21
import org.apache.commons.lang3.tuple.Triple;
22
import org.cyclops.commoncapabilities.api.ingredient.IngredientComponent;
23
import org.cyclops.integrateddynamics.api.evaluate.EvaluationException;
24
import org.cyclops.integrateddynamics.api.evaluate.operator.IOperator;
25
import org.cyclops.integrateddynamics.api.evaluate.variable.*;
26
import org.cyclops.integrateddynamics.api.ingredient.IIngredientComponentHandler;
27
import org.cyclops.integrateddynamics.api.logicprogrammer.IConfigRenderPattern;
28
import org.cyclops.integrateddynamics.core.evaluate.build.OperatorBuilder;
29
import org.cyclops.integrateddynamics.core.evaluate.operator.IterativeFunction;
30
import org.cyclops.integrateddynamics.core.evaluate.operator.OperatorBase;
31
import org.cyclops.integrateddynamics.core.evaluate.variable.*;
32
import org.cyclops.integrateddynamics.core.helper.L10NValues;
33
import org.cyclops.integrateddynamics.core.ingredient.IngredientComponentHandlers;
34

35
import javax.annotation.Nullable;
36
import java.util.Arrays;
37
import java.util.List;
38
import java.util.Optional;
39
import java.util.concurrent.Callable;
40
import java.util.stream.Collectors;
41
import java.util.stream.Stream;
42

43
/**
44
 * Collection of operator builders.
45
 * @author rubensworks
46
 */
47
public class OperatorBuilders {
×
48

49
    // --------------- Logical builders ---------------
50
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> LOGICAL = OperatorBuilder.forType(ValueTypes.BOOLEAN).appendKind("logical");
5✔
51
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> LOGICAL_1_PREFIX = LOGICAL.inputTypes(1, ValueTypes.BOOLEAN).renderPattern(IConfigRenderPattern.PREFIX_1);
7✔
52
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> LOGICAL_2 = LOGICAL.inputTypes(2, ValueTypes.BOOLEAN).renderPattern(IConfigRenderPattern.INFIX);
7✔
53

54
    // --------------- Value propagators ---------------
55
    public static final IOperatorValuePropagator<Integer, IValue> PROPAGATOR_INTEGER_VALUE = ValueTypeInteger.ValueInteger::of;
2✔
56
    public static final IOperatorValuePropagator<Long, IValue> PROPAGATOR_LONG_VALUE = ValueTypeLong.ValueLong::of;
2✔
57
    public static final IOperatorValuePropagator<Boolean, IValue> PROPAGATOR_BOOLEAN_VALUE = ValueTypeBoolean.ValueBoolean::of;
2✔
58
    public static final IOperatorValuePropagator<Double, IValue> PROPAGATOR_DOUBLE_VALUE = ValueTypeDouble.ValueDouble::of;
2✔
59
    public static final IOperatorValuePropagator<String, IValue> PROPAGATOR_STRING_VALUE = ValueTypeString.ValueString::of;
2✔
60
    public static final IOperatorValuePropagator<Optional<Tag>, IValue> PROPAGATOR_NBT_VALUE = ValueTypeNbt.ValueNbt::of;
2✔
61
    public static final IOperatorValuePropagator<Optional<CompoundTag>, IValue> PROPAGATOR_NBT_COMPOUND_VALUE = opt -> ValueTypeNbt.ValueNbt.of(opt.map(t -> (Tag) t));
9✔
62
    public static final IOperatorValuePropagator<ResourceLocation, ValueTypeString.ValueString> PROPAGATOR_RESOURCELOCATION_MODNAME = resourceLocation -> {
2✔
63
        String modId = resourceLocation.getNamespace();
3✔
64
        Optional<? extends ModContainer> mod = ModList.get().getModContainerById(modId);
4✔
65
        String modName = mod
2✔
66
                .map(modContainer -> modContainer.getModInfo().getDisplayName())
6✔
67
                .orElse("Minecraft");
3✔
68
        return ValueTypeString.ValueString.of(modName);
3✔
69
    };
70

71
    // --------------- Arithmetic builders ---------------
72
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ARITHMETIC = OperatorBuilder.forType(ValueTypes.CATEGORY_NUMBER).appendKind("arithmetic").conditionalOutputTypeDeriver((operator, input) -> {
7✔
73
        IValueType[] original = ValueHelpers.from(input);
3✔
74
        IValueTypeNumber[] types = new IValueTypeNumber[original.length];
4✔
75
        for(int i = 0; i < original.length; i++) {
8✔
76
            if (original[i].isCategory()) {
5!
77
                // This avoids a class-cast exception in cases where we don't know the exact type.
78
                return original[i];
×
79
            }
80
            types[i] = (IValueTypeNumber) original[i];
7✔
81
        }
82
        return ValueTypes.CATEGORY_NUMBER.getLowestType(types);
4✔
83
    });
84
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ARITHMETIC_1_SUFFIX = ARITHMETIC.inputTypes(1, ValueTypes.CATEGORY_NUMBER).renderPattern(IConfigRenderPattern.SUFFIX_1);
7✔
85
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ARITHMETIC_2 = ARITHMETIC.inputTypes(2, ValueTypes.CATEGORY_NUMBER).renderPattern(IConfigRenderPattern.INFIX);
7✔
86
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ARITHMETIC_2_PREFIX = ARITHMETIC.inputTypes(2, ValueTypes.CATEGORY_NUMBER).renderPattern(IConfigRenderPattern.PREFIX_2);
7✔
87

88
    // --------------- Integer builders ---------------
89
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> INTEGER = OperatorBuilder.forType(ValueTypes.INTEGER).appendKind("integer");
5✔
90
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> INTEGER_1_SUFFIX = INTEGER.inputTypes(1, ValueTypes.INTEGER).renderPattern(IConfigRenderPattern.SUFFIX_1);
7✔
91
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> INTEGER_2 = INTEGER.inputTypes(2, ValueTypes.INTEGER).renderPattern(IConfigRenderPattern.INFIX);
7✔
92

93
    // --------------- Relational builders ---------------
94
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> RELATIONAL = OperatorBuilder.forType(ValueTypes.BOOLEAN).appendKind("relational");
5✔
95
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> RELATIONAL_2 = RELATIONAL.inputTypes(2, ValueTypes.INTEGER).renderPattern(IConfigRenderPattern.INFIX);
7✔
96

97
    // --------------- Binary builders ---------------
98
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> BINARY = OperatorBuilder.forType(ValueTypes.INTEGER).appendKind("binary");
5✔
99
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> BINARY_1_PREFIX = BINARY.inputTypes(1, ValueTypes.INTEGER).renderPattern(IConfigRenderPattern.PREFIX_1);
7✔
100
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> BINARY_2 = BINARY.inputTypes(2, ValueTypes.INTEGER).renderPattern(IConfigRenderPattern.INFIX);
7✔
101

102
    // --------------- String builders ---------------
103
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> STRING = OperatorBuilder.forType(ValueTypes.STRING).appendKind("string");
5✔
104
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> STRING_1_PREFIX = STRING.inputTypes(1, ValueTypes.STRING).renderPattern(IConfigRenderPattern.PREFIX_1);
7✔
105
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> STRING_2 = STRING.inputTypes(2, ValueTypes.STRING).renderPattern(IConfigRenderPattern.INFIX);
7✔
106
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> STRING_2_LONG = STRING.inputTypes(2, ValueTypes.STRING).renderPattern(IConfigRenderPattern.INFIX_LONG);
7✔
107

108
    // --------------- Double builders ---------------
109
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> DOUBLE = OperatorBuilder.forType(ValueTypes.DOUBLE).appendKind("double");
5✔
110
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> DOUBLE_1_PREFIX = DOUBLE.inputTypes(1, ValueTypes.DOUBLE).renderPattern(IConfigRenderPattern.PREFIX_1);
7✔
111
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> DOUBLE_2 = DOUBLE.inputTypes(2, ValueTypes.DOUBLE).renderPattern(IConfigRenderPattern.INFIX);
7✔
112

113
    // --------------- Number builders ---------------
114
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> NUMBER = OperatorBuilder.forType(ValueTypes.CATEGORY_NUMBER).appendKind("number");
5✔
115
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> NUMBER_1_PREFIX = NUMBER.inputTypes(1, ValueTypes.CATEGORY_NUMBER).renderPattern(IConfigRenderPattern.PREFIX_1);
7✔
116
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> NUMBER_1_LONG = NUMBER.inputTypes(1, ValueTypes.CATEGORY_NUMBER).renderPattern(IConfigRenderPattern.SUFFIX_1_LONG);
7✔
117

118
    // --------------- Nullable builders ---------------
119
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> NULLABLE = OperatorBuilder.forType(ValueTypes.CATEGORY_NULLABLE).appendKind("general");
5✔
120
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> NULLABLE_1_PREFIX = NULLABLE.inputTypes(1, ValueTypes.CATEGORY_NULLABLE).renderPattern(IConfigRenderPattern.PREFIX_1);
7✔
121

122
    // --------------- List builders ---------------
123
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> LIST = OperatorBuilder.forType(ValueTypes.LIST).appendKind("list");
5✔
124
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> LIST_1_PREFIX = LIST.inputTypes(1, ValueTypes.LIST).renderPattern(IConfigRenderPattern.PREFIX_1);
7✔
125

126
    // --------------- Block builders ---------------
127
    public static final OperatorBuilder BLOCK = OperatorBuilder.forType(ValueTypes.OBJECT_BLOCK).appendKind("block");
5✔
128
    public static final OperatorBuilder BLOCK_1_SUFFIX_LONG = BLOCK.inputTypes(1, ValueTypes.OBJECT_BLOCK).renderPattern(IConfigRenderPattern.SUFFIX_1_LONG);
7✔
129
    public static final OperatorBuilder BLOCK_INFIX_VERYLONG = BLOCK.inputTypes(2, ValueTypes.OBJECT_BLOCK).renderPattern(IConfigRenderPattern.INFIX_VERYLONG);
7✔
130
    public static final IOperatorValuePropagator<OperatorBase.SafeVariablesGetter, Optional<SoundType>> BLOCK_SOUND = input -> {
2✔
131
        ValueObjectTypeBlock.ValueBlock block = input.getValue(0, ValueTypes.OBJECT_BLOCK);
6✔
132
        if(block.getRawValue().isPresent()) {
4!
133
            return Optional.of(block.getRawValue().get().getSoundType());
7✔
134
        }
135
        return Optional.empty();
×
136
    };
137

138
    // --------------- ItemStack builders ---------------
139
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ITEMSTACK = OperatorBuilder.forType(ValueTypes.OBJECT_ITEMSTACK).appendKind("itemstack");
5✔
140
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ITEMSTACK_1_PREFIX_LONG = ITEMSTACK.inputTypes(1, ValueTypes.OBJECT_ITEMSTACK).renderPattern(IConfigRenderPattern.PREFIX_1_LONG);
7✔
141
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ITEMSTACK_1_SUFFIX_LONG = ITEMSTACK.inputTypes(1, ValueTypes.OBJECT_ITEMSTACK).renderPattern(IConfigRenderPattern.SUFFIX_1_LONG);
7✔
142
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ITEMSTACK_2 = ITEMSTACK.inputTypes(2, ValueTypes.OBJECT_ITEMSTACK).renderPattern(IConfigRenderPattern.INFIX);
7✔
143
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ITEMSTACK_2_LONG = ITEMSTACK.inputTypes(2, ValueTypes.OBJECT_ITEMSTACK).renderPattern(IConfigRenderPattern.INFIX_LONG);
7✔
144
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ITEMSTACK_1_INTEGER_1 = ITEMSTACK.inputTypes(new IValueType[]{ValueTypes.OBJECT_ITEMSTACK, ValueTypes.INTEGER}).renderPattern(IConfigRenderPattern.INFIX);
15✔
145
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ITEMSTACK_3 = ITEMSTACK.inputTypes(3, ValueTypes.OBJECT_ITEMSTACK).renderPattern(IConfigRenderPattern.PREFIX_3_LONG);
7✔
146
    public static final IterativeFunction.PrePostBuilder<ItemStack, IValue> FUNCTION_ITEMSTACK = IterativeFunction.PrePostBuilder.begin()
2✔
147
            .appendPre(input -> {
2✔
148
                ValueObjectTypeItemStack.ValueItemStack value = input.getValue(0, ValueTypes.OBJECT_ITEMSTACK);
6✔
149
                return value.getRawValue();
3✔
150
            });
151
    public static final IterativeFunction.PrePostBuilder<ItemStack, Integer> FUNCTION_ITEMSTACK_TO_INT =
2✔
152
            FUNCTION_ITEMSTACK.appendPost(PROPAGATOR_INTEGER_VALUE);
2✔
153
    public static final IterativeFunction.PrePostBuilder<ItemStack, Boolean> FUNCTION_ITEMSTACK_TO_BOOLEAN =
2✔
154
            FUNCTION_ITEMSTACK.appendPost(PROPAGATOR_BOOLEAN_VALUE);
2✔
155

156
    public static final IterativeFunction.PrePostBuilder<IEnergyStorage, IValue> FUNCTION_ENERGYSTORAGEITEM = IterativeFunction.PrePostBuilder.begin()
2✔
157
            .appendPre(input -> {
2✔
158
                ValueObjectTypeItemStack.ValueItemStack a = input.getValue(0, ValueTypes.OBJECT_ITEMSTACK);
6✔
159
                if(!a.getRawValue().isEmpty()) {
4!
160
                    return a.getRawValue().getCapability(Capabilities.EnergyStorage.ITEM);
6✔
161
                }
162
                return null;
×
163
            });
164
    public static final IterativeFunction.PrePostBuilder<IEnergyStorage, Integer> FUNCTION_CONTAINERITEM_TO_INT =
2✔
165
            FUNCTION_ENERGYSTORAGEITEM.appendPost(org.cyclops.integrateddynamics.core.evaluate.OperatorBuilders.PROPAGATOR_INTEGER_VALUE);
2✔
166
    public static final IterativeFunction.PrePostBuilder<IEnergyStorage, Boolean> FUNCTION_CONTAINERITEM_TO_BOOLEAN =
2✔
167
            FUNCTION_ENERGYSTORAGEITEM.appendPost(org.cyclops.integrateddynamics.core.evaluate.OperatorBuilders.PROPAGATOR_BOOLEAN_VALUE);
2✔
168

169
    // --------------- Entity builders ---------------
170
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ENTITY = OperatorBuilder.forType(ValueTypes.OBJECT_ENTITY).appendKind("entity");
5✔
171
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ENTITY_1_SUFFIX = ENTITY.inputTypes(1, ValueTypes.OBJECT_ENTITY).renderPattern(IConfigRenderPattern.SUFFIX_1);
7✔
172
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ENTITY_1_SUFFIX_LONG = ENTITY.inputTypes(1, ValueTypes.OBJECT_ENTITY).renderPattern(IConfigRenderPattern.SUFFIX_1_LONG);
7✔
173
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ENTITY_1_ITEMSTACK_1 = ENTITY.inputTypes(new IValueType[]{ValueTypes.OBJECT_ENTITY, ValueTypes.OBJECT_ITEMSTACK}).renderPattern(IConfigRenderPattern.INFIX_LONG);
15✔
174
    public static final IterativeFunction.PrePostBuilder<Entity, IValue> FUNCTION_ENTITY = IterativeFunction.PrePostBuilder.begin()
2✔
175
            .appendPre(input -> {
2✔
176
                ValueObjectTypeEntity.ValueEntity a = input.getValue(0, ValueTypes.OBJECT_ENTITY);
6✔
177
                return a.getRawValue().isPresent() ? a.getRawValue().get() : null;
10!
178
            });
179
    public static final IterativeFunction.PrePostBuilder<Entity, Double> FUNCTION_ENTITY_TO_DOUBLE =
2✔
180
            FUNCTION_ENTITY.appendPost(PROPAGATOR_DOUBLE_VALUE);
2✔
181
    public static final IterativeFunction.PrePostBuilder<Entity, Boolean> FUNCTION_ENTITY_TO_BOOLEAN =
2✔
182
            FUNCTION_ENTITY.appendPost(PROPAGATOR_BOOLEAN_VALUE);
2✔
183

184
    // --------------- FluidStack builders ---------------
185
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> FLUIDSTACK = OperatorBuilder.forType(ValueTypes.OBJECT_FLUIDSTACK).appendKind("fluidstack");
5✔
186
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> FLUIDSTACK_1_SUFFIX_LONG = FLUIDSTACK.inputTypes(1, ValueTypes.OBJECT_FLUIDSTACK).renderPattern(IConfigRenderPattern.SUFFIX_1_LONG);
7✔
187
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> FLUIDSTACK_2 = FLUIDSTACK.inputTypes(2, ValueTypes.OBJECT_FLUIDSTACK).renderPattern(IConfigRenderPattern.INFIX);
7✔
188
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> FLUIDSTACK_2_LONG = FLUIDSTACK.inputTypes(2, ValueTypes.OBJECT_FLUIDSTACK).renderPattern(IConfigRenderPattern.INFIX_LONG);
7✔
189
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> FLUIDSTACK_3 = ITEMSTACK.inputTypes(3, ValueTypes.OBJECT_FLUIDSTACK).renderPattern(IConfigRenderPattern.PREFIX_3_LONG);
7✔
190
    public static final IterativeFunction.PrePostBuilder<FluidStack, IValue> FUNCTION_FLUIDSTACK = IterativeFunction.PrePostBuilder.begin()
2✔
191
            .appendPre(input -> {
2✔
192
                ValueObjectTypeFluidStack.ValueFluidStack a = input.getValue(0, ValueTypes.OBJECT_FLUIDSTACK);
6✔
193
                return a.getRawValue();
3✔
194
            });
195
    public static final IterativeFunction.PrePostBuilder<FluidStack, Integer> FUNCTION_FLUIDSTACK_TO_INT =
2✔
196
            FUNCTION_FLUIDSTACK.appendPost(PROPAGATOR_INTEGER_VALUE);
2✔
197
    public static final IterativeFunction.PrePostBuilder<FluidStack, Boolean> FUNCTION_FLUIDSTACK_TO_BOOLEAN =
2✔
198
            FUNCTION_FLUIDSTACK.appendPost(PROPAGATOR_BOOLEAN_VALUE);
2✔
199

200
    // --------------- Operator builders ---------------
201
    public static final IterativeFunction.PrePostBuilder<Pair<IOperator, OperatorBase.SafeVariablesGetter>, IValue> FUNCTION_OPERATOR_TAKE_OPERATOR = IterativeFunction.PrePostBuilder.begin()
2✔
202
            .appendPre(input -> {
2✔
203
                IOperator innerOperator = input.getValue(0, ValueTypes.OPERATOR).getRawValue();
7✔
204
                if (input.getVariables().length > 1) {
5✔
205
                    if (innerOperator.getRequiredInputLength() == 1) {
4✔
206
                        IValue applyingValue = input.getValue(1);
4✔
207
                        MutableComponent error = innerOperator.validateTypes(new IValueType[]{applyingValue.getType()});
10✔
208
                        if (error != null) {
2✔
209
                            throw new EvaluationException(error);
5✔
210
                        }
211
                    } else if (innerOperator.getRequiredInputLength() > 0) {
4!
212
                        if (!ValueHelpers.correspondsTo(input.getVariables()[1].getType(), innerOperator.getInputTypes()[0])) {
11!
213
                            MutableComponent error = Component.translatable(L10NValues.OPERATOR_ERROR_WRONGCURRYINGTYPE,
×
214
                                    Component.translatable(innerOperator.getTranslationKey()),
×
215
                                    Component.translatable(input.getVariables()[1].getType().getTranslationKey()),
×
216
                                    0,
×
217
                                    Component.translatable(innerOperator.getInputTypes()[0].getTranslationKey())
×
218
                            );
219
                            throw new EvaluationException(error);
×
220
                        }
221
                    }
222
                }
223
                return Pair.<IOperator, OperatorBase.SafeVariablesGetter>of(innerOperator,
7✔
224
                        new OperatorBase.SafeVariablesGetter.Shifted(1, input.getVariables()));
2✔
225
            });
226
    public static final IterativeFunction.PrePostBuilder<IOperator, IValue> FUNCTION_ONE_OPERATOR = IterativeFunction.PrePostBuilder.begin()
2✔
227
            .appendPre(input -> getSafeOperator(input.getValue(0, ValueTypes.OPERATOR), ValueTypes.CATEGORY_ANY));
10✔
228
    public static final IterativeFunction.PrePostBuilder<IOperator, IValue> FUNCTION_ONE_PREDICATE = IterativeFunction.PrePostBuilder.begin()
2✔
229
            .appendPre(input -> getSafePredictate(input.getValue(0, ValueTypes.OPERATOR)));
9✔
230
    public static final IterativeFunction.PrePostBuilder<Pair<IOperator, IOperator>, IValue> FUNCTION_TWO_OPERATORS = IterativeFunction.PrePostBuilder.begin()
2✔
231
            .appendPre(input -> {
2✔
232
                IOperator second = getSafeOperator(input.getValue(1, ValueTypes.OPERATOR), ValueTypes.CATEGORY_ANY);
8✔
233
                IValueType[] secondInputs = second.getInputTypes();
3✔
234
                if(secondInputs.length < 1) {
4!
235
                    throw new EvaluationException(Component.translatable(
×
236
                            L10NValues.OPERATOR_ERROR_OPERATORPARAMWRONGINPUTLENGTH,
237
                            1, second.getLocalizedNameFull(), secondInputs.length));
×
238
                }
239
                IValueType secondInputType = secondInputs[0];
4✔
240
                if (ValueHelpers.correspondsTo(secondInputType, ValueTypes.OPERATOR)) {
4✔
241
                    secondInputType = ValueTypes.CATEGORY_ANY;
2✔
242
                }
243
                IOperator first = getSafeOperator(input.getValue(0, ValueTypes.OPERATOR), secondInputType);
8✔
244
                return Pair.of(first, second);
4✔
245
            });
246
    public static final IterativeFunction.PrePostBuilder<Pair<IOperator, IOperator>, IValue> FUNCTION_TWO_PREDICATES = IterativeFunction.PrePostBuilder.begin()
2✔
247
            .appendPre(input -> {
2✔
248
                IOperator first = getSafePredictate(input.getValue(0, ValueTypes.OPERATOR));
7✔
249
                IOperator second = getSafePredictate(input.getValue(1, ValueTypes.OPERATOR));
7✔
250
                return Pair.of(first, second);
4✔
251
            });
252
    public static final IterativeFunction.PrePostBuilder<Triple<IOperator, IOperator, IOperator>, IValue> FUNCTION_THREE_OPERATORS = IterativeFunction.PrePostBuilder.begin()
2✔
253
            .appendPre(input -> {
2✔
254
                IOperator third = getSafeOperator(input.getValue(2, ValueTypes.OPERATOR), ValueTypes.CATEGORY_ANY);
8✔
255
                IValueType<?>[] types = third.getInputTypes();
3✔
256
                if(types.length < 2) {
4!
257
                    throw new EvaluationException(Component.translatable(
×
258
                            L10NValues.OPERATOR_ERROR_OPERATORPARAMWRONGINPUTLENGTH,
259
                            2, third.getLocalizedNameFull(), types.length));
×
260
                }
261
                IValueType<?> firstOutputType = types[0];
4✔
262
                IValueType<?> secondOutputType = types[1];
4✔
263
                if (ValueHelpers.correspondsTo(firstOutputType, ValueTypes.OPERATOR)) {
4✔
264
                    firstOutputType = ValueTypes.CATEGORY_ANY;
2✔
265
                }
266
                if (ValueHelpers.correspondsTo(secondOutputType, ValueTypes.OPERATOR)) {
4✔
267
                    secondOutputType = ValueTypes.CATEGORY_ANY;
2✔
268
                }
269
                IOperator first = getSafeOperator(input.getValue(0, ValueTypes.OPERATOR), firstOutputType);
8✔
270
                IOperator second = getSafeOperator(input.getValue(1, ValueTypes.OPERATOR), secondOutputType);
8✔
271
                return Triple.of(first, second, third);
5✔
272
            });
273
    public static final IterativeFunction.PrePostBuilder<Pair<IOperator, OperatorBase.SafeVariablesGetter>, IValue> FUNCTION_OPERATOR_TAKE_OPERATOR_LIST = IterativeFunction.PrePostBuilder.begin()
2✔
274
            .appendPre(input -> {
2✔
275
                ValueTypeOperator.ValueOperator valueOperator = input.getValue(0, ValueTypes.OPERATOR);
6✔
276
                IOperator innerOperator = valueOperator.getRawValue();
3✔
277
                input.getValue(1, ValueTypes.LIST); // To trigger exception on invalid type
5✔
278
                return Pair.<IOperator, OperatorBase.SafeVariablesGetter>of(innerOperator,
7✔
279
                        new OperatorBase.SafeVariablesGetter.Shifted(1, input.getVariables()));
2✔
280
            });
281
    /**
282
     * Corresponds to {@link ValueHelpers#evaluateOperator(IOperator, IVariable[])}.
283
     */
284
    public static OperatorBuilder.IConditionalOutputTypeDeriver OPERATOR_CONDITIONAL_OUTPUT_DERIVER = (operator, variablesAll) -> {
2✔
285
        try {
286
            IValue value = variablesAll[0].getValue();
5✔
287
            // In some cases, validation can succeed because of parameters being ANY.
288
            // In this case, return a dummy type.
289
            if (!(value instanceof ValueTypeOperator.ValueOperator)) {
3✔
290
                return ValueTypes.CATEGORY_ANY;
2✔
291
            }
292
            IOperator innerOperator = ((ValueTypeOperator.ValueOperator) value).getRawValue();
4✔
293
            IVariable[] variables = ArrayUtils.subarray(variablesAll, 1, variablesAll.length);
7✔
294
            int requiredLength = innerOperator.getRequiredInputLength();
3✔
295
            if (requiredLength == variables.length) {
4✔
296
                Component error = innerOperator.validateTypes(ValueHelpers.from(variables));
5✔
297
                if (error != null) {
2!
298
                    return innerOperator.getOutputType();
×
299
                }
300
                return innerOperator.getConditionalOutputType(variables);
4✔
301
            } else {
302
                if (variables.length > requiredLength) { // We have MORE variables as input than the operator accepts
4!
303
                    IVariable[] acceptableVariables = ArrayUtils.subarray(variables, 0, requiredLength);
6✔
304
                    IVariable[] remainingVariables = ArrayUtils.subarray(variables, requiredLength, variables.length);
7✔
305

306
                    // Pass all required variables to the operator, and forward all remaining ones to the resulting operator
307
                    IValue result = ValueHelpers.evaluateOperator(innerOperator, acceptableVariables);
4✔
308

309
                    // Error if the result is NOT an operator
310
                    if (result.getType() != ValueTypes.OPERATOR) {
4!
311
                        throw new EvaluationException(Component.translatable(L10NValues.OPERATOR_ERROR_CURRYINGOVERFLOW,
×
312
                                Component.translatable(innerOperator.getTranslationKey()),
×
313
                                requiredLength,
×
314
                                variables.length,
×
315
                                Component.translatable(result.getType().getTranslationKey())));
×
316
                    }
317

318
                    // Pass all remaining variables to the resulting operator
319
                    IOperator nextOperator = ((ValueTypeOperator.ValueOperator) result).getRawValue();
4✔
320
                    Component error = nextOperator.validateTypes(ValueHelpers.from(remainingVariables));
5✔
321
                    if (error != null) {
2!
322
                        return nextOperator.getOutputType();
×
323
                    }
324
                    return nextOperator.getConditionalOutputType(remainingVariables);
4✔
325
                } else {
326
                    return ValueTypes.OPERATOR;
×
327
                }
328
            }
329
        } catch (EvaluationException e) {
×
330
            return ValueTypes.CATEGORY_ANY;
×
331
        }
332
    };
333
    /**
334
     * Corresponds to {@link ValueHelpers#evaluateOperator(IOperator, IVariable[])}, but with the input variable being a list.
335
     */
336
    public static OperatorBuilder.IConditionalOutputTypeDeriver OPERATOR_CONDITIONAL_OUTPUT_DERIVER_LIST = (operator, variablesIn) -> {
2✔
337
        try {
338
            // Get second param as list
339
            IValue valueList = variablesIn[1].getValue();
5✔
340
            // In some cases, validation can succeed because of parameters being ANY.
341
            // In this case, return a dummy type.
342
            if (!(valueList instanceof ValueTypeList.ValueList)) {
3!
343
                return ValueTypes.CATEGORY_ANY;
×
344
            }
345
            IValueTypeListProxy listProxy = ((ValueTypeList.ValueList) valueList).getRawValue();
4✔
346

347
            // Expand the list to a variable array, with variablesIn[0] prepended (the operator to apply).
348
            IVariable[] variablesExpanded = Stream.concat(
5✔
349
                    Stream.of(variablesIn[0]),
3✔
350
                    Stream.<IValue>of(Iterables.toArray(listProxy, IValue.class))
4✔
351
                        .map(Variable::new)
1✔
352
            ).toArray(IVariable[]::new);
6✔
353

354
            return OPERATOR_CONDITIONAL_OUTPUT_DERIVER.getConditionalOutputType(operator, variablesExpanded);
5✔
355
        } catch (EvaluationException e) {
×
356
            return ValueTypes.CATEGORY_ANY;
×
357
        }
358
    };
359
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> OPERATOR = OperatorBuilder
1✔
360
            .forType(ValueTypes.OPERATOR).appendKind("operator");
4✔
361
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> OPERATOR_2_INFIX_LONG = OPERATOR
11✔
362
            .inputTypes(new IValueType[]{ValueTypes.OPERATOR, ValueTypes.CATEGORY_ANY})
2✔
363
            .renderPattern(IConfigRenderPattern.INFIX);
2✔
364
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> OPERATOR_1_PREFIX_LONG = OPERATOR
7✔
365
            .inputTypes(new IValueType[]{ValueTypes.OPERATOR})
2✔
366
            .renderPattern(IConfigRenderPattern.PREFIX_1_LONG);
2✔
367

368
    // --------------- String builders ---------------
369

370
    public static final IterativeFunction.PrePostBuilder<ResourceLocation, IValue> FUNCTION_STRING_TO_RESOURCE_LOCATION = IterativeFunction.PrePostBuilder.begin()
2✔
371
            .appendPre(input -> {
2✔
372
                ValueTypeString.ValueString a = input.getValue(0, ValueTypes.STRING);
6✔
373
                return ValueHelpers.createResourceLocationInEvaluation(a.getRawValue());
4✔
374
            });
375

376
    // --------------- Operator helpers ---------------
377

378
    /**
379
     * Get the operator from a value in a safe manner.
380
     * @param value The operator value.
381
     * @param expectedOutput The expected output value type.
382
     * @return The operator.
383
     * @throws EvaluationException If the operator is not a predicate.
384
     */
385
    public static IOperator getSafeOperator(ValueTypeOperator.ValueOperator value, IValueType expectedOutput) throws EvaluationException {
386
        IOperator operator = value.getRawValue();
3✔
387
        if (!ValueHelpers.correspondsTo(operator.getOutputType(), expectedOutput)) {
5✔
388
            MutableComponent error = Component.translatable(L10NValues.OPERATOR_ERROR_ILLEGALPROPERY,
8✔
389
                    Component.translatable(expectedOutput.getTranslationKey()),
6✔
390
                    Component.translatable(operator.getOutputType().getTranslationKey()),
7✔
391
                    operator.getLocalizedNameFull());
2✔
392
            throw new EvaluationException(error);
5✔
393
        }
394
        return operator;
2✔
395
    }
396

397
    /**
398
     * Get the predicate from a value in a safe manner.
399
     * It is expected that the operator returns a boolean.
400
     * @param value The operator value.
401
     * @return The operator.
402
     * @throws EvaluationException If the operator is not a predicate.
403
     */
404
    public static IOperator getSafePredictate(ValueTypeOperator.ValueOperator value) throws EvaluationException {
405
        return getSafeOperator(value, ValueTypes.BOOLEAN);
4✔
406
    }
407

408
    /**
409
     * Create a type validator for operator operator type validators.
410
     * @param expectedSubTypes The expected types that must be present in the operator (not including the first
411
     *                         operator type itself.
412
     * @return The type validator instance.
413
     */
414
    public static OperatorBuilder.ITypeValidator createOperatorTypeValidator(final IValueType<?>... expectedSubTypes) {
415
        final int subOperatorLength = expectedSubTypes.length;
3✔
416
        final Component expected = Component.translatable(
3✔
417
                org.cyclops.integrateddynamics.core.helper.Helpers.createPatternOfLength(subOperatorLength), (Object[]) ValueHelpers.from(expectedSubTypes));
4✔
418
        return (operator, input) -> {
4✔
419
            if (input.length == 0 || !ValueHelpers.correspondsTo(input[0], ValueTypes.OPERATOR)) {
9✔
420
                String givenName = input.length == 0 ? "null" : input[0].getTranslationKey();
10✔
421
                return Component.translatable(L10NValues.VALUETYPE_ERROR_INVALIDOPERATOROPERATOR,
8✔
422
                        0, givenName);
6✔
423
            }
424
            if (input.length != subOperatorLength + 1) {
6✔
425
                IValueType<?>[] operatorInputs = Arrays.copyOfRange(input, 1, input.length);
7✔
426
                Component given = Component.translatable(
4✔
427
                        org.cyclops.integrateddynamics.core.helper.Helpers.createPatternOfLength(operatorInputs.length), (Object[]) ValueHelpers.from(operatorInputs));
4✔
428
                return Component.translatable(L10NValues.VALUETYPE_ERROR_INVALIDOPERATORSIGNATURE,
13✔
429
                        expected, given);
430
            }
431
            return null;
2✔
432
        };
433
    }
434

435
    // --------------- NBT builders ---------------
436
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> NBT = OperatorBuilder.forType(ValueTypes.NBT).appendKind("nbt");
5✔
437
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> NBT_1_PREFIX_LONG = NBT.inputTypes(ValueTypes.NBT).renderPattern(IConfigRenderPattern.PREFIX_1_LONG);
11✔
438
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> NBT_1_SUFFIX_LONG = NBT.inputTypes(ValueTypes.NBT).renderPattern(IConfigRenderPattern.SUFFIX_1_LONG);
11✔
439
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> NBT_2 = NBT.inputTypes(ValueTypes.NBT, ValueTypes.STRING).renderPattern(IConfigRenderPattern.INFIX_LONG);
15✔
440
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> NBT_2_NBT = NBT.inputTypes(ValueTypes.NBT, ValueTypes.NBT).renderPattern(IConfigRenderPattern.INFIX_LONG);
15✔
441
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> NBT_3 = NBT.inputTypes(ValueTypes.NBT, ValueTypes.STRING, ValueTypes.STRING).output(ValueTypes.NBT).renderPattern(IConfigRenderPattern.INFIX_2_LONG);
21✔
442

443
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, IValue> FUNCTION_NBT = IterativeFunction.PrePostBuilder.begin()
2✔
444
            .appendPre(input -> {
2✔
445
                ValueTypeNbt.ValueNbt value = input.getValue(0, ValueTypes.NBT);
6✔
446
                return value.getRawValue();
3✔
447
            });
448
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, IValue> FUNCTION_NBT_COMPOUND_ENTRY = IterativeFunction.PrePostBuilder.begin()
2✔
449
            .appendPre(input -> {
2✔
450
                ValueTypeNbt.ValueNbt valueNbt = input.getValue(0, ValueTypes.NBT);
6✔
451
                ValueTypeString.ValueString valueString = input.getValue(1, ValueTypes.STRING);
6✔
452
                return valueNbt.getRawValue()
4✔
453
                        .filter(tag -> tag instanceof CompoundTag)
6✔
454
                        .map(tag -> ((CompoundTag) tag).get(valueString.getRawValue()));
7✔
455
            });
456
    public static final IterativeFunction.PrePostBuilder<Triple<Optional<CompoundTag>, String, OperatorBase.SafeVariablesGetter>, IValue> FUNCTION_NBT_COPY_FOR_VALUE = IterativeFunction.PrePostBuilder.begin()
2✔
457
            .appendPre(input -> {
2✔
458
                ValueTypeNbt.ValueNbt valueNbt = input.getValue(0, ValueTypes.NBT);
6✔
459
                ValueTypeString.ValueString valueString = input.getValue(1, ValueTypes.STRING);
6✔
460
                return Triple.of(valueNbt.getRawValue()
5✔
461
                                .filter(t -> t instanceof CompoundTag)
5✔
462
                                .map(t -> (CompoundTag) t)
5✔
463
                                .map(CompoundTag::copy), valueString.getRawValue(),
7✔
464
                        new OperatorBase.SafeVariablesGetter.Shifted(2, input.getVariables()));
2✔
465
            });
466

467
    public static final IOperatorValuePropagator<Optional<Tag>, Optional<CompoundTag>> PROPAGATOR_NBT_COMPOUND = opt -> opt
5✔
468
            .filter(t -> t instanceof CompoundTag)
5✔
469
            .map(t -> (CompoundTag) t);
4✔
470

471
    public static final IterativeFunction.PrePostBuilder<Optional<CompoundTag>, IValue> FUNCTION_NBT_COMPOUND = FUNCTION_NBT.appendPre(PROPAGATOR_NBT_COMPOUND);
4✔
472

473
    public static final IterativeFunction.PrePostBuilder<Optional<CompoundTag>, Integer> FUNCTION_NBT_COMPOUND_TO_INT =
2✔
474
            FUNCTION_NBT_COMPOUND.appendPost(PROPAGATOR_INTEGER_VALUE);
2✔
475
    public static final IterativeFunction.PrePostBuilder<Optional<CompoundTag>, Boolean> FUNCTION_NBT_COMPOUND_TO_BOOLEAN =
2✔
476
            FUNCTION_NBT_COMPOUND.appendPost(PROPAGATOR_BOOLEAN_VALUE);
2✔
477

478
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, Integer> FUNCTION_NBT_COMPOUND_ENTRY_TO_INT =
2✔
479
            FUNCTION_NBT_COMPOUND_ENTRY.appendPost(PROPAGATOR_INTEGER_VALUE);
2✔
480
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, Long> FUNCTION_NBT_COMPOUND_ENTRY_TO_LONG =
2✔
481
            FUNCTION_NBT_COMPOUND_ENTRY.appendPost(PROPAGATOR_LONG_VALUE);
2✔
482
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, Double> FUNCTION_NBT_COMPOUND_ENTRY_TO_DOUBLE =
2✔
483
            FUNCTION_NBT_COMPOUND_ENTRY.appendPost(PROPAGATOR_DOUBLE_VALUE);
2✔
484
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, Boolean> FUNCTION_NBT_COMPOUND_ENTRY_TO_BOOLEAN =
2✔
485
            FUNCTION_NBT_COMPOUND_ENTRY.appendPost(PROPAGATOR_BOOLEAN_VALUE);
2✔
486
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, String> FUNCTION_NBT_COMPOUND_ENTRY_TO_STRING =
2✔
487
            FUNCTION_NBT_COMPOUND_ENTRY.appendPost(PROPAGATOR_STRING_VALUE);
2✔
488
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, Optional<Tag>> FUNCTION_NBT_COMPOUND_ENTRY_TO_NBT =
2✔
489
            FUNCTION_NBT_COMPOUND_ENTRY.appendPost(PROPAGATOR_NBT_VALUE);
2✔
490

491
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, Integer> FUNCTION_NBT_TO_INT =
2✔
492
            FUNCTION_NBT.appendPost(PROPAGATOR_INTEGER_VALUE);
2✔
493
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, Long> FUNCTION_NBT_TO_LONG =
2✔
494
            FUNCTION_NBT.appendPost(PROPAGATOR_LONG_VALUE);
2✔
495
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, Double> FUNCTION_NBT_TO_DOUBLE =
2✔
496
            FUNCTION_NBT.appendPost(PROPAGATOR_DOUBLE_VALUE);
2✔
497
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, Boolean> FUNCTION_NBT_TO_BOOLEAN =
2✔
498
            FUNCTION_NBT.appendPost(PROPAGATOR_BOOLEAN_VALUE);
2✔
499
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, String> FUNCTION_NBT_TO_STRING =
2✔
500
            FUNCTION_NBT.appendPost(PROPAGATOR_STRING_VALUE);
2✔
501
    public static final IterativeFunction.PrePostBuilder<Optional<Tag>, Optional<Tag>> FUNCTION_NBT_TO_NBT =
2✔
502
            FUNCTION_NBT.appendPost(PROPAGATOR_NBT_VALUE);
2✔
503

504
    public static final IterativeFunction.PrePostBuilder<Triple<Optional<CompoundTag>, String, OperatorBase.SafeVariablesGetter>, Optional<CompoundTag>>
505
            FUNCTION_NBT_COPY_FOR_VALUE_TO_NBT = FUNCTION_NBT_COPY_FOR_VALUE.appendPost(PROPAGATOR_NBT_COMPOUND_VALUE);
4✔
506

507
    // --------------- Ingredients builders ---------------
508
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> INGREDIENTS = OperatorBuilder.forType(ValueTypes.OBJECT_INGREDIENTS).appendKind("ingredients");
5✔
509
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> INGREDIENTS_1_PREFIX_LONG = INGREDIENTS
7✔
510
            .inputTypes(ValueTypes.OBJECT_INGREDIENTS).renderPattern(IConfigRenderPattern.SUFFIX_1_LONG);
4✔
511
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> INGREDIENTS_3_ITEMSTACK = INGREDIENTS
15✔
512
            .inputTypes(ValueTypes.OBJECT_INGREDIENTS, ValueTypes.INTEGER, ValueTypes.OBJECT_ITEMSTACK)
2✔
513
            .renderPattern(IConfigRenderPattern.INFIX_2_LONG).output(ValueTypes.OBJECT_INGREDIENTS);
4✔
514
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> INGREDIENTS_3_FLUIDSTACK = INGREDIENTS
15✔
515
            .inputTypes(ValueTypes.OBJECT_INGREDIENTS, ValueTypes.INTEGER, ValueTypes.OBJECT_FLUIDSTACK)
2✔
516
            .renderPattern(IConfigRenderPattern.INFIX_2_LONG).output(ValueTypes.OBJECT_INGREDIENTS);
4✔
517
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> INGREDIENTS_3_LONG = INGREDIENTS
15✔
518
            .inputTypes(ValueTypes.OBJECT_INGREDIENTS, ValueTypes.INTEGER, ValueTypes.LONG)
2✔
519
            .renderPattern(IConfigRenderPattern.INFIX_2_LONG).output(ValueTypes.OBJECT_INGREDIENTS);
4✔
520
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> INGREDIENTS_2_LIST = INGREDIENTS
11✔
521
            .inputTypes(ValueTypes.OBJECT_INGREDIENTS, ValueTypes.LIST)
2✔
522
            .renderPattern(IConfigRenderPattern.INFIX_LONG).output(ValueTypes.OBJECT_INGREDIENTS);
4✔
523

524
    public static OperatorBase.IFunction createFunctionIngredientsList(Callable<IngredientComponent<?, ?>> componentReference) {
525
        return variables -> {
3✔
526
            IngredientComponent<?, ?> component = null;
2✔
527
            try {
528
                component = componentReference.call();
4✔
529
            } catch (Exception e) {
×
530
                e.printStackTrace();
×
531
            }
1✔
532
            IIngredientComponentHandler componentHandler = IngredientComponentHandlers.REGISTRY.getComponentHandler(component);
4✔
533
            ValueObjectTypeIngredients.ValueIngredients value = variables.getValue(0, ValueTypes.OBJECT_INGREDIENTS);
6✔
534
            List<?> list = Lists.newArrayList();
2✔
535
            if (value.getRawValue().isPresent()) {
4!
536
                list = value.getRawValue().get().getInstances(component);
7✔
537
            }
538
            return ValueTypeList.ValueList.ofList(componentHandler.getValueType(), list.stream()
8✔
539
                    .map(i -> componentHandler.toValue(i)).collect(Collectors.toList()));
8✔
540
        };
541
    }
542

543
    public static <VT extends IValueType<V>, V extends IValue, T, M> List<T> unwrapIngredientComponentList(IngredientComponent<T, M> component,
544
                                                                                                           ValueTypeList.ValueList<VT, V> list)
545
            throws EvaluationException {
546
        IIngredientComponentHandler<VT, V, T, M> componentHandler = IngredientComponentHandlers.REGISTRY.getComponentHandler(component);
4✔
547
        if (list.getRawValue().getValueType() != componentHandler.getValueType()) {
6!
548
            throw new EvaluationException(Component.translatable(
×
549
                    L10NValues.VALUETYPE_ERROR_INVALIDLISTVALUETYPE,
550
                    Component.translatable(componentHandler.getValueType().getTranslationKey()),
×
551
                    Component.translatable(list.getRawValue().getValueType().getTranslationKey())));
×
552
        }
553
        List<T> listTransformed = Lists.newArrayListWithExpectedSize(list.getRawValue().getLength());
5✔
554
        for (V value : list.getRawValue()) {
11✔
555
            listTransformed.add(componentHandler.toInstance(value));
6✔
556
        }
1✔
557
        return listTransformed;
2✔
558
    }
559

560
    // --------------- Recipe builders ---------------
561
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> RECIPE = OperatorBuilder.forType(ValueTypes.OBJECT_RECIPE).appendKind("recipe");
5✔
562
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> RECIPE_1_SUFFIX_LONG = RECIPE
7✔
563
            .inputTypes(ValueTypes.OBJECT_RECIPE).renderPattern(IConfigRenderPattern.SUFFIX_1_LONG);
4✔
564
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> RECIPE_2_INFIX = RECIPE
11✔
565
            .inputTypes(ValueTypes.OBJECT_RECIPE, ValueTypes.OBJECT_INGREDIENTS)
2✔
566
            .renderPattern(IConfigRenderPattern.INFIX_LONG);
2✔
567
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> RECIPE_2_PREFIX = RECIPE
12✔
568
            .inputTypes(ValueTypes.OBJECT_INGREDIENTS, ValueTypes.OBJECT_INGREDIENTS)
2✔
569
            .renderPattern(IConfigRenderPattern.PREFIX_2_LONG);
2✔
570

571
    // --------------- Capability helpers ---------------
572

573
    /**
574
     * Helper function to create an operator function builder for deriving capabilities from an itemstack.
575
     * @param capabilityReference The capability instance reference.
576
     * @param <T> The capability type.
577
     * @param <C> The capability context type.
578
     * @return The builder.
579
     */
580
    public static <T, C> IterativeFunction.PrePostBuilder<T, IValue> getItemCapability(@Nullable final ICapabilityReference<T, C> capabilityReference) {
581
        return IterativeFunction.PrePostBuilder.begin()
×
582
                .appendPre(input -> {
×
583
                    ValueObjectTypeItemStack.ValueItemStack a = input.getValue(0, ValueTypes.OBJECT_ITEMSTACK);
×
584
                    if(!a.getRawValue().isEmpty()) {
×
585
                        return a.getRawValue().getCapability(capabilityReference.getReference(), null);
×
586
                    }
587
                    return null;
×
588
                });
589
    }
590

591
    public static interface ICapabilityReference<T, C> {
592
        public ItemCapability<T, C> getReference();
593
    }
594

595
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc