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

CyclopsMC / IntegratedDynamics / 20210191346

14 Dec 2025 03:32PM UTC coverage: 19.514% (-33.5%) from 53.061%
20210191346

push

github

rubensworks
Remove deprecations

663 of 8728 branches covered (7.6%)

Branch coverage included in aggregate %.

6786 of 29445 relevant lines covered (23.05%)

1.09 hits per line

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

76.81
/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.fluids.FluidStack;
18
import net.neoforged.neoforge.transfer.access.ItemAccess;
19
import net.neoforged.neoforge.transfer.energy.EnergyHandler;
20
import org.apache.commons.lang3.ArrayUtils;
21
import org.apache.commons.lang3.tuple.Pair;
22
import org.apache.commons.lang3.tuple.Triple;
23
import org.cyclops.commoncapabilities.api.ingredient.IngredientComponent;
24
import org.cyclops.integrateddynamics.api.evaluate.EvaluationException;
25
import org.cyclops.integrateddynamics.api.evaluate.operator.IOperator;
26
import org.cyclops.integrateddynamics.api.evaluate.variable.*;
27
import org.cyclops.integrateddynamics.api.ingredient.IIngredientComponentHandler;
28
import org.cyclops.integrateddynamics.api.logicprogrammer.IConfigRenderPattern;
29
import org.cyclops.integrateddynamics.core.evaluate.build.OperatorBuilder;
30
import org.cyclops.integrateddynamics.core.evaluate.operator.IterativeFunction;
31
import org.cyclops.integrateddynamics.core.evaluate.operator.OperatorBase;
32
import org.cyclops.integrateddynamics.core.evaluate.variable.*;
33
import org.cyclops.integrateddynamics.core.helper.L10NValues;
34
import org.cyclops.integrateddynamics.core.ingredient.IngredientComponentHandlers;
35

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

159
    public static final IterativeFunction.PrePostBuilder<EnergyHandler, IValue> FUNCTION_ENERGYSTORAGEITEM = IterativeFunction.PrePostBuilder.begin()
2✔
160
            .appendPre(input -> {
2✔
161
                ValueObjectTypeItemStack.ValueItemStack a = input.getValue(0, ValueTypes.OBJECT_ITEMSTACK);
×
162
                if(!a.getRawValue().isEmpty()) {
×
163
                    return ItemAccess.forStack(a.getRawValue()).getCapability(Capabilities.Energy.ITEM);
×
164
                }
165
                return null;
×
166
            });
167
    public static final IterativeFunction.PrePostBuilder<EnergyHandler, Long> FUNCTION_CONTAINERITEM_TO_LONG =
2✔
168
            FUNCTION_ENERGYSTORAGEITEM.appendPost(OperatorBuilders.PROPAGATOR_LONG_VALUE);
2✔
169
    public static final IterativeFunction.PrePostBuilder<EnergyHandler, Boolean> FUNCTION_CONTAINERITEM_TO_BOOLEAN =
2✔
170
            FUNCTION_ENERGYSTORAGEITEM.appendPost(OperatorBuilders.PROPAGATOR_BOOLEAN_VALUE);
2✔
171

172
    // --------------- Entity builders ---------------
173
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ENTITY = OperatorBuilder.forType(ValueTypes.OBJECT_ENTITY).appendKind("entity");
5✔
174
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ENTITY_1_SUFFIX = ENTITY.inputTypes(1, ValueTypes.OBJECT_ENTITY).renderPattern(IConfigRenderPattern.SUFFIX_1);
7✔
175
    public static final OperatorBuilder<OperatorBase.SafeVariablesGetter> ENTITY_1_SUFFIX_LONG = ENTITY.inputTypes(1, ValueTypes.OBJECT_ENTITY).renderPattern(IConfigRenderPattern.SUFFIX_1_LONG);
7✔
176
    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✔
177
    public static final IterativeFunction.PrePostBuilder<Entity, IValue> FUNCTION_ENTITY = IterativeFunction.PrePostBuilder.begin()
2✔
178
            .appendPre(input -> {
2✔
179
                ValueObjectTypeEntity.ValueEntity a = input.getValue(0, ValueTypes.OBJECT_ENTITY);
×
180
                return a.getRawValue().isPresent() ? a.getRawValue().get() : null;
×
181
            });
182
    public static final IterativeFunction.PrePostBuilder<Entity, Double> FUNCTION_ENTITY_TO_DOUBLE =
2✔
183
            FUNCTION_ENTITY.appendPost(PROPAGATOR_DOUBLE_VALUE);
2✔
184
    public static final IterativeFunction.PrePostBuilder<Entity, Boolean> FUNCTION_ENTITY_TO_BOOLEAN =
2✔
185
            FUNCTION_ENTITY.appendPost(PROPAGATOR_BOOLEAN_VALUE);
2✔
186

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

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

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

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

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

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

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

372
    // --------------- String builders ---------------
373

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

380
    // --------------- Operator helpers ---------------
381

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

575
    // --------------- Capability helpers ---------------
576

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

595
    public static interface ICapabilityReference<T, C> {
596
        public ItemCapability<T, C> getReference();
597
    }
598

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