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

CyclopsMC / IntegratedDynamics / 19267223570

11 Nov 2025 01:30PM UTC coverage: 53.046% (+0.04%) from 53.006%
19267223570

push

github

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

2875 of 8772 branches covered (32.77%)

Branch coverage included in aggregate %.

17354 of 29363 relevant lines covered (59.1%)

3.07 hits per line

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

87.06
/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 OperatorBuilder<OperatorBase.SafeVariablesGetter> FLUIDSTACK_1_PREFIX_LONG = FLUIDSTACK.inputTypes(1, ValueTypes.OBJECT_FLUIDSTACK).renderPattern(IConfigRenderPattern.PREFIX_1_LONG);
7✔
191
    public static final IterativeFunction.PrePostBuilder<FluidStack, IValue> FUNCTION_FLUIDSTACK = IterativeFunction.PrePostBuilder.begin()
2✔
192
            .appendPre(input -> {
2✔
193
                ValueObjectTypeFluidStack.ValueFluidStack a = input.getValue(0, ValueTypes.OBJECT_FLUIDSTACK);
6✔
194
                return a.getRawValue();
3✔
195
            });
196
    public static final IterativeFunction.PrePostBuilder<FluidStack, Integer> FUNCTION_FLUIDSTACK_TO_INT =
2✔
197
            FUNCTION_FLUIDSTACK.appendPost(PROPAGATOR_INTEGER_VALUE);
2✔
198
    public static final IterativeFunction.PrePostBuilder<FluidStack, Boolean> FUNCTION_FLUIDSTACK_TO_BOOLEAN =
2✔
199
            FUNCTION_FLUIDSTACK.appendPost(PROPAGATOR_BOOLEAN_VALUE);
2✔
200

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

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

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

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

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

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

369
    // --------------- String builders ---------------
370

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

377
    // --------------- Operator helpers ---------------
378

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

572
    // --------------- Capability helpers ---------------
573

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

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

596
}
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