minecraft-src/net/minecraft/world/item/enchantment/LevelBasedValue.java
2025-07-04 01:41:11 +03:00

170 lines
6.1 KiB
Java

package net.minecraft.world.item.enchantment;
import com.mojang.datafixers.util.Either;
import com.mojang.serialization.Codec;
import com.mojang.serialization.DataResult;
import com.mojang.serialization.MapCodec;
import com.mojang.serialization.codecs.RecordCodecBuilder;
import java.util.List;
import net.minecraft.core.Registry;
import net.minecraft.core.registries.BuiltInRegistries;
import net.minecraft.util.Mth;
public interface LevelBasedValue {
Codec<LevelBasedValue> DISPATCH_CODEC = BuiltInRegistries.ENCHANTMENT_LEVEL_BASED_VALUE_TYPE
.byNameCodec()
.dispatch(LevelBasedValue::codec, mapCodec -> mapCodec);
Codec<LevelBasedValue> CODEC = Codec.either(LevelBasedValue.Constant.CODEC, DISPATCH_CODEC)
.xmap(
either -> either.map(constant -> constant, levelBasedValue -> levelBasedValue),
levelBasedValue -> levelBasedValue instanceof LevelBasedValue.Constant constant ? Either.left(constant) : Either.right(levelBasedValue)
);
static MapCodec<? extends LevelBasedValue> bootstrap(Registry<MapCodec<? extends LevelBasedValue>> registry) {
Registry.register(registry, "clamped", LevelBasedValue.Clamped.CODEC);
Registry.register(registry, "fraction", LevelBasedValue.Fraction.CODEC);
Registry.register(registry, "levels_squared", LevelBasedValue.LevelsSquared.CODEC);
Registry.register(registry, "linear", LevelBasedValue.Linear.CODEC);
return Registry.register(registry, "lookup", LevelBasedValue.Lookup.CODEC);
}
static LevelBasedValue.Constant constant(float value) {
return new LevelBasedValue.Constant(value);
}
static LevelBasedValue.Linear perLevel(float base, float perLevelAfterFirst) {
return new LevelBasedValue.Linear(base, perLevelAfterFirst);
}
static LevelBasedValue.Linear perLevel(float perLevel) {
return perLevel(perLevel, perLevel);
}
static LevelBasedValue.Lookup lookup(List<Float> values, LevelBasedValue fallback) {
return new LevelBasedValue.Lookup(values, fallback);
}
float calculate(int level);
MapCodec<? extends LevelBasedValue> codec();
public record Clamped(LevelBasedValue value, float min, float max) implements LevelBasedValue {
public static final MapCodec<LevelBasedValue.Clamped> CODEC = RecordCodecBuilder.<LevelBasedValue.Clamped>mapCodec(
instance -> instance.group(
LevelBasedValue.CODEC.fieldOf("value").forGetter(LevelBasedValue.Clamped::value),
Codec.FLOAT.fieldOf("min").forGetter(LevelBasedValue.Clamped::min),
Codec.FLOAT.fieldOf("max").forGetter(LevelBasedValue.Clamped::max)
)
.apply(instance, LevelBasedValue.Clamped::new)
)
.validate(
clamped -> clamped.max <= clamped.min
? DataResult.error(() -> "Max must be larger than min, min: " + clamped.min + ", max: " + clamped.max)
: DataResult.success(clamped)
);
@Override
public float calculate(int level) {
return Mth.clamp(this.value.calculate(level), this.min, this.max);
}
@Override
public MapCodec<LevelBasedValue.Clamped> codec() {
return CODEC;
}
}
public record Constant(float value) implements LevelBasedValue {
public static final Codec<LevelBasedValue.Constant> CODEC = Codec.FLOAT.xmap(LevelBasedValue.Constant::new, LevelBasedValue.Constant::value);
public static final MapCodec<LevelBasedValue.Constant> TYPED_CODEC = RecordCodecBuilder.mapCodec(
instance -> instance.group(Codec.FLOAT.fieldOf("value").forGetter(LevelBasedValue.Constant::value)).apply(instance, LevelBasedValue.Constant::new)
);
@Override
public float calculate(int level) {
return this.value;
}
@Override
public MapCodec<LevelBasedValue.Constant> codec() {
return TYPED_CODEC;
}
}
public record Fraction(LevelBasedValue numerator, LevelBasedValue denominator) implements LevelBasedValue {
public static final MapCodec<LevelBasedValue.Fraction> CODEC = RecordCodecBuilder.mapCodec(
instance -> instance.group(
LevelBasedValue.CODEC.fieldOf("numerator").forGetter(LevelBasedValue.Fraction::numerator),
LevelBasedValue.CODEC.fieldOf("denominator").forGetter(LevelBasedValue.Fraction::denominator)
)
.apply(instance, LevelBasedValue.Fraction::new)
);
@Override
public float calculate(int level) {
float f = this.denominator.calculate(level);
return f == 0.0F ? 0.0F : this.numerator.calculate(level) / f;
}
@Override
public MapCodec<LevelBasedValue.Fraction> codec() {
return CODEC;
}
}
public record LevelsSquared(float added) implements LevelBasedValue {
public static final MapCodec<LevelBasedValue.LevelsSquared> CODEC = RecordCodecBuilder.mapCodec(
instance -> instance.group(Codec.FLOAT.fieldOf("added").forGetter(LevelBasedValue.LevelsSquared::added)).apply(instance, LevelBasedValue.LevelsSquared::new)
);
@Override
public float calculate(int level) {
return Mth.square(level) + this.added;
}
@Override
public MapCodec<LevelBasedValue.LevelsSquared> codec() {
return CODEC;
}
}
public record Linear(float base, float perLevelAboveFirst) implements LevelBasedValue {
public static final MapCodec<LevelBasedValue.Linear> CODEC = RecordCodecBuilder.mapCodec(
instance -> instance.group(
Codec.FLOAT.fieldOf("base").forGetter(LevelBasedValue.Linear::base),
Codec.FLOAT.fieldOf("per_level_above_first").forGetter(LevelBasedValue.Linear::perLevelAboveFirst)
)
.apply(instance, LevelBasedValue.Linear::new)
);
@Override
public float calculate(int level) {
return this.base + this.perLevelAboveFirst * (level - 1);
}
@Override
public MapCodec<LevelBasedValue.Linear> codec() {
return CODEC;
}
}
public record Lookup(List<Float> values, LevelBasedValue fallback) implements LevelBasedValue {
public static final MapCodec<LevelBasedValue.Lookup> CODEC = RecordCodecBuilder.mapCodec(
instance -> instance.group(
Codec.FLOAT.listOf().fieldOf("values").forGetter(LevelBasedValue.Lookup::values),
LevelBasedValue.CODEC.fieldOf("fallback").forGetter(LevelBasedValue.Lookup::fallback)
)
.apply(instance, LevelBasedValue.Lookup::new)
);
@Override
public float calculate(int level) {
return level <= this.values.size() ? (Float)this.values.get(level - 1) : this.fallback.calculate(level);
}
@Override
public MapCodec<LevelBasedValue.Lookup> codec() {
return CODEC;
}
}
}