minecraft-src/net/minecraft/world/item/enchantment/ItemEnchantments.java
2025-07-04 03:45:38 +03:00

157 lines
5.4 KiB
Java

package net.minecraft.world.item.enchantment;
import com.mojang.serialization.Codec;
import it.unimi.dsi.fastutil.objects.Object2IntOpenHashMap;
import it.unimi.dsi.fastutil.objects.Object2IntMap.Entry;
import java.util.Collections;
import java.util.Optional;
import java.util.Set;
import java.util.function.Consumer;
import java.util.function.Predicate;
import net.minecraft.core.Holder;
import net.minecraft.core.HolderLookup;
import net.minecraft.core.HolderSet;
import net.minecraft.core.Registry;
import net.minecraft.core.HolderSet.Named;
import net.minecraft.core.component.DataComponentGetter;
import net.minecraft.core.registries.Registries;
import net.minecraft.network.RegistryFriendlyByteBuf;
import net.minecraft.network.chat.Component;
import net.minecraft.network.codec.ByteBufCodecs;
import net.minecraft.network.codec.StreamCodec;
import net.minecraft.resources.ResourceKey;
import net.minecraft.tags.EnchantmentTags;
import net.minecraft.tags.TagKey;
import net.minecraft.world.item.Item;
import net.minecraft.world.item.TooltipFlag;
import net.minecraft.world.item.component.TooltipProvider;
import org.jetbrains.annotations.Nullable;
public class ItemEnchantments implements TooltipProvider {
public static final ItemEnchantments EMPTY = new ItemEnchantments(new Object2IntOpenHashMap<>());
private static final Codec<Integer> LEVEL_CODEC = Codec.intRange(1, 255);
public static final Codec<ItemEnchantments> CODEC = Codec.unboundedMap(Enchantment.CODEC, LEVEL_CODEC)
.xmap(map -> new ItemEnchantments(new Object2IntOpenHashMap<>(map)), itemEnchantments -> itemEnchantments.enchantments);
public static final StreamCodec<RegistryFriendlyByteBuf, ItemEnchantments> STREAM_CODEC = StreamCodec.composite(
ByteBufCodecs.map(Object2IntOpenHashMap::new, Enchantment.STREAM_CODEC, ByteBufCodecs.VAR_INT),
itemEnchantments -> itemEnchantments.enchantments,
ItemEnchantments::new
);
final Object2IntOpenHashMap<Holder<Enchantment>> enchantments;
ItemEnchantments(Object2IntOpenHashMap<Holder<Enchantment>> enchantments) {
this.enchantments = enchantments;
for (Entry<Holder<Enchantment>> entry : enchantments.object2IntEntrySet()) {
int i = entry.getIntValue();
if (i < 0 || i > 255) {
throw new IllegalArgumentException("Enchantment " + entry.getKey() + " has invalid level " + i);
}
}
}
public int getLevel(Holder<Enchantment> enchantment) {
return this.enchantments.getInt(enchantment);
}
@Override
public void addToTooltip(Item.TooltipContext context, Consumer<Component> tooltipAdder, TooltipFlag flag, DataComponentGetter componentGetter) {
HolderLookup.Provider provider = context.registries();
HolderSet<Enchantment> holderSet = getTagOrEmpty(provider, Registries.ENCHANTMENT, EnchantmentTags.TOOLTIP_ORDER);
for (Holder<Enchantment> holder : holderSet) {
int i = this.enchantments.getInt(holder);
if (i > 0) {
tooltipAdder.accept(Enchantment.getFullname(holder, i));
}
}
for (Entry<Holder<Enchantment>> entry : this.enchantments.object2IntEntrySet()) {
Holder<Enchantment> holder2 = (Holder<Enchantment>)entry.getKey();
if (!holderSet.contains(holder2)) {
tooltipAdder.accept(Enchantment.getFullname((Holder<Enchantment>)entry.getKey(), entry.getIntValue()));
}
}
}
private static <T> HolderSet<T> getTagOrEmpty(@Nullable HolderLookup.Provider registries, ResourceKey<Registry<T>> registryKey, TagKey<T> key) {
if (registries != null) {
Optional<Named<T>> optional = registries.lookupOrThrow(registryKey).get(key);
if (optional.isPresent()) {
return (HolderSet<T>)optional.get();
}
}
return HolderSet.direct();
}
public Set<Holder<Enchantment>> keySet() {
return Collections.unmodifiableSet(this.enchantments.keySet());
}
public Set<Entry<Holder<Enchantment>>> entrySet() {
return Collections.unmodifiableSet(this.enchantments.object2IntEntrySet());
}
public int size() {
return this.enchantments.size();
}
public boolean isEmpty() {
return this.enchantments.isEmpty();
}
public boolean equals(Object object) {
if (this == object) {
return true;
} else {
return object instanceof ItemEnchantments itemEnchantments ? this.enchantments.equals(itemEnchantments.enchantments) : false;
}
}
public int hashCode() {
return this.enchantments.hashCode();
}
public String toString() {
return "ItemEnchantments{enchantments=" + this.enchantments + "}";
}
public static class Mutable {
private final Object2IntOpenHashMap<Holder<Enchantment>> enchantments = new Object2IntOpenHashMap<>();
public Mutable(ItemEnchantments enchantments) {
this.enchantments.putAll(enchantments.enchantments);
}
public void set(Holder<Enchantment> enchantment, int level) {
if (level <= 0) {
this.enchantments.removeInt(enchantment);
} else {
this.enchantments.put(enchantment, Math.min(level, 255));
}
}
public void upgrade(Holder<Enchantment> enchantment, int level) {
if (level > 0) {
this.enchantments.merge(enchantment, Math.min(level, 255), Integer::max);
}
}
public void removeIf(Predicate<Holder<Enchantment>> predicate) {
this.enchantments.keySet().removeIf(predicate);
}
public int getLevel(Holder<Enchantment> enchantment) {
return this.enchantments.getOrDefault(enchantment, 0);
}
public Set<Holder<Enchantment>> keySet() {
return this.enchantments.keySet();
}
public ItemEnchantments toImmutable() {
return new ItemEnchantments(this.enchantments);
}
}
}